Mirrorr 2010-07-27
[size=x-small;]目录是存储于磁盘上的数据结构,它由许多目录项组成。每个目录项描述了一个指向其它目录或文件的入口。目录的作用是把文件组织成层次结构,便于管理。[/size]
[size=x-small;]
9.2.1工作目录
每一个进程都有一个当前目录和进程相关,获取当前目录用getcwd函数得到。其原型包含在unistd.h头文件中:
char * getcwd(char *buffer, size_t size);
函数getcwd返回用C语言字符串表示的当前目录名。参数buffer和size分别表示你分配的内存地址和长度,存放当前目录名之用。也可以给出buffer为null,size为0,表示让getcwd函数自动分配内存存放目录名。
由于事先不知道目录名的长度会是多少,所以,下面的例子先分配一个适当大小的内存区域,如果不够大,重新分配一个两倍大的内存区域,直至成功。
char * new_getcwd() { int size=100; char *buffer=(char *)xmalloc(size); while(1) { char *value=getcwd(buffer,size); if(value!=NULL) return buffer; free(buffer); size*=2; buffer=(char *)xmalloc(size); } }
改变当前目录的方法是用函数chdir,其原型如下:
int chdir(const char* filename);
其中filename是要改变的当前目录。成功返回0,出错返回-1。
9.2.2操作目录结构
文件系统的目录就象是文件一样,只是存储的内容是目录项。可以用opendir函数打开目录项,然后读取其内容。该函数包含在dirent.h头文件中:
DIR * opendir( const char *dirname);
dirname是要打开的目录名称(完全路径名或相对路径名)。调用成功返回一个指向DIR数据结构的指针。DIR结构描述了打开的目录需要的所有参数,
比如当前读取的目录项序号等。和FILE*结构相似,对目录中目录项的读写要引用DIR*。从打开的目录中读取目录项用readdir函数,其原型如下:
struct dirent * readdir( DIR *dirstream);
参数dirstream就是opendir函数返回的指向DIR结构的指针。成功调用返回指向下一个目录项数据结构的指针。数据结构dirent描述了一个目录项的信息(包括该目录项描述的文件或目录的信息),它的结构如下:
char d_name[],目录或文件的名称。它是一个0结尾的字符串(ASCIIZ字符串)。 ino_t d_fileno,该数据结构包含了文件的信息,可以用stat函数更详细的解析。 unsigned char d_namlen,文件或目录名称的长度,不包括结尾的0。 unsigned char d_type,文件或目录的类型。它有可能的取值如下: DT_UNKNOWN,未知的类型 DT_REG,普通文件 DT_DIR,普通目录 DT_FIFO,命名管道或FIFO DT_SOCK,本地套接口 DT_CHR,字符设备文件 DT_BLK,块设备文件
函数readdir的成功调用不仅返回指向dirent的指针,而且使DIR结构中的当前目录项指针指向下一个位置。函数调用不成功,返回null指针。
目录使用完毕用closedir函数关闭;
int closedir(DIR *dirstream);
该函数成功返回0,失败返回-1。
下面的程序列出当前的文件名,和ls命令相似:
#include <stddef.h> #include <stdio.h> #include <sys/types.h> #include <dirent.h> int main(void) { DIR *dp; struct dirent *ep; dp=opendir("./"); if(dp!=NULL) { while(ep = readdir(dp)) puts(ep->d_name); closedir(dp); } else puts("Couldn't open the directory .\n"); return 0; }
9.2.3目录、文件的属性
读取文件的属性有三个函数可以实现,它们都返回结构stat,它是在sys/stat.h头文件中定义的,它描述读取的文件的属性。下面介绍structstat的成员:
mode_tst_mode,它描述了文件的属性,包括类型和权限位。为测试这些属性予定义了专门的宏。测试文件类型,用下面的宏:
int S_ISDIR(mode_t m):如果文件是目录,返回非0,否则返回0。 int S_ISCHR(mode_t m):如果文件是字符设备文件,返回非0,否则返回0。 int S_ISBLK(mode_t m):如果文件是块设备,返回非0,否则返回0。 int S_ISREG(mode_t m):如果文件是普通文件,返回非0,否则返回0。 int S_ISFIFO(mode_t m):如果文件是FIFO,返回非0,否则返回0。 int S_ISLNK(mode_t m):如果文件是符号连接,返回非0,否则返回0。 int S_ISSOCK(mode_t m):如果文件是UNIX套接口,返回非0,否则返回0。 ino_t st_ino,文件的inode号,它唯一决定同一设备上的某个文件。 dev_t st_dev,文件所在的设备号。 nlink_t st_nlink,连接到同一个文件上的目录项数目。这里指的是硬连接,不是符号连接。如果这个数为0,文件系统将自动删除该文件。 uid_t st_uid,文件的user ID。 gid_t st_gid,文件的group ID。 off_t st_size,普通文件的长度。如果是特殊设备文件,该项没有意义。如果是符号连接,它实际上是连接到的文件的长度。 time_t st_atime,最近一次操作文件的时间。time_t是UNIX中表示时间的常用方法。它是从1970年1月1日零时起的秒数。它也称为Coordinated Universal Time,在GNU系统中,是无符号长整数。 unsigned long int st_atime_usec,是最近一次操作文件的时间的小数部分。 time_t st_mtime,最近一次修改文件内容的时间。 unsigned long int st_mtime_usec,最近一次修改文件内容的时间的小数部分。 time_t st_ctime,最近一次修改文件属性的时间。 unsigned long int st_ctime_usec,最近一次修改文件属性的时间的小数部分。 blkcnt_t st_blocks,文件实际占用的磁盘的块数。块的长度是512字节。这和文件的长度可能不同,有两个原因:系统可能用某些空间存储文件的管理信息;另一个原因是前面讲过的有可能包含“空洞”,即连续的0。 unsigned int st_blksize,文件读写操作中使用的块的大小。可以根据这个值设置缓冲区的大小。
上面就是structstat的内容。取得文件的属性,有下面三个函数:
int stat(const char *filename, struct stat *buf); int fstat(int filedes, struct stat *buf); int lstat(const char *filename, struct stat *buf);
上面的三个函数都是取得文件的属性存放到buf中。filename是文件的名字。stat和lstat的差别是,stat读取符号连接时,要读取被连接
的文件的属性;而lstat读取连接本身的属性,并不对连接进行跟踪。fstat和stat的差别是要提供打开的文件描述符,而不是文件名。
这三个函数成功都返回0,失败返回-1。
9.2.4文件的其它操作
建立文件的硬连接,用link函数,它的原型在头文件unistd.h中:
int link(const char *oldname, const char *newname);
建立newname到oldname的连接。
如果建立符号连接,用syslink函数:
int syslink(const char *oldname, const char *newname);
上面两个函数,成功返回0,失败返回-1。
删除文件用unlink系统调用:
int unlink(const char *filename);
它只是删除文件名到文件的连接,如果文件的连接计数等于0,则系统删除文件。
改变文件名称,用rename函数:
int rename(const char *oldname, const char *newname);
创建目录:
int mkdir(const char *filename, mode_t mode);
9.2.5一个例子
下面举一个新例子,它删除当前目录中所有时间晚于当前时间1天的文件。
#include <stddef.h> #include <stdio.h> #include <sys/types.h> #include <sys/stat.h> #include <dirent.h> #include <time.h> int main(void) { DIR *dp; struct dirent *ep; struct stat st; dp=opendir("./"); if(dp!=NULL) { while(ep = readdir(dp)) { if(ep->d_name[0]!='.') { stat(ep->d_name,&st); if (time(NULL)-st.st_mtime>24*3600)&S_ISREG(st.st_mode)) { printf("file %s will be deleted\n",ep->d_name); unlink(ep->d_name); } else { printf("file %s will be reserved\n",ep->d_name); } } } closedir(dp); } else puts("Couldn't open the directory.\n"); return 0; }
本文转自:http://www.eefocus.com/article/07-10/26799s.html