目录操作

[size=x-small;]目录是存储于磁盘上的数据结构,它由许多目录项组成。每个目录项描述了一个指向其它目录或文件的入口。目录的作用是把文件组织成层次结构,便于管理。[/size]
[size=x-small;]


9.2.1[b] 工作目录[/b]

每一个进程都有一个当前目录和进程相关,获取当前目录用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 [b]操作目录结构[/b]

文件系统的目录就象是文件一样,只是存储的内容是目录项。可以用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 [b]目录、文件的属性[/b]

读取文件的属性有三个函数可以实现,它们都返回结构stat,它是在sys/stat.h头文件中定义的,它描述读取的文件的属性。下面介绍struct stat的成员:

mode_t st_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,文件读写操作中使用的块的大小。可以根据这个值设置缓冲区的大小。


上面就是struct stat的内容。取得文件的属性,有下面三个函数:

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 [b]文件的其它操作[/b]

建立文件的硬连接,用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 [b]一个例子[/b]

下面举一个新例子,它删除当前目录中所有时间晚于当前时间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;
}



本文转自:[url]http://www.eefocus.com/article/07-10/26799s.html[/url]
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值