目录操作

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(&quot;./&quot;);
      if(dp!=NULL)
      {
         while(ep = readdir(dp))
             puts(ep-&gt;d_name);
         closedir(dp);
       }
       else
           puts(&quot;Couldn't open the directory .\n&quot;);
       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(&quot;./&quot;);
    if(dp!=NULL)
    {
        while(ep = readdir(dp))
        {
            if(ep-&gt;d_name[0]!='.')
            { 
                stat(ep-&gt;d_name,&amp;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

相关推荐

NewActivity / 0评论 2020-05-08