Linux系统编程——基于文件指针的文件操作(缓冲)

目录操作
  • 创建和删除目录:

原型为:

#include <sys/stat.h> #include <sys/types.h>

#include <unistd.h> int mkdir(const char *pathname, mode_t mode); //创建目录,mode是目录权限

int rmdir(const char *pathname); //删除目录

  • 获取目录信息:

原型为:

#include <sys/types.h>

#include <dirent.h> DIR *opendir(const char *name); //打开一个目录

struct dirent *readdir(DIR *dir); //读取目录的一项信息,并返回该项信息的结构体指针

void rewinddir(DIR *dir); //重新定位到目录文件的头部

void seekdir(DIR *dir,off_t offset);//用来设置目录流目前的读取位置

off_t telldir(DIR *dir); //返回目录流当前的读取位置

int closedir(DIR *dir); //关闭目录文件


#include <sys/types.h>

#include <sys/stat.h>

#include <unistd.h>

int stat(const char *pathname, struct stat *buf); //获取文件状态

读取目录信息的步骤为:

 用opendir函数打开目录;

 使用readdir函数迭代读取目录的内容,如果已经读取到目录末尾,又想重新开始读,则可以使用rewinddir函数将文件指针重新定位到目录文件的起始位置;

 用closedir函数关闭目录

opendir()用来打开参数name指定的目录,并返回DIR*形态的目录流,和文件操作函数open()类似,接下来对

目录的读取和搜索都要使用此返回值。函数失败则返回NULL;

readdir()函数用来读取目录的信息,并返回一个结构体指针,该指针保存了目录的相关信息。有错误发生或

者读取到目录文件尾则返回NULL;

dirent结构体如下:

struct dirent
{
    ino_t d_ino; /* inode number(此目录进入点的inode) */
    off_t d_off; /* offset to the next dirent(目录开头到进入点的位移 */
    unsigned short d_reclen; /* length of this record(目录名的长度) */
    unsigned char d_type; /* type of file(所指的文件类型) */
    char d_name[256]; /* filename(文件名) */
};

seekdir()函数用来设置目录流目前的读取位置,再调用readdir()函数时,便可以从此新位置开始读取。参数offset代表距离目录文件开头的偏移量。
telldir()函数用来返回目录流当前的读取位置。
结构体stat的定义为:

struct stat {
    dev_t st_dev; /*如果是设备,返回设备表述符,否则为0*/
    ino_t st_ino; /* i节点号 */
    mode_t st_mode; /* 文件类型 */
    nlink_t st_nlink; /* 链接数 */
    uid_t st_uid; /* 属主ID */
    gid_t st_gid; /* 组ID */
    dev_t st_rdev; /* 设备类型*/
    off_t st_size; /* 文件大小,字节表示 */
    blksize_t st_blksize; /* 块大小*/
    blkcnt_t st_blocks; /* 块数 */
    time_t st_atime; /* 最后访问时间*/
    time_t st_mtime; /* 最后修改时间*/
    time_t st_ctime; /* 最后权限修改时间 */
};

ex1: opendir()函数测试:

#include <func.h>

int main(int argc,char* argv[])
{
	ARGS_CHECK(argc,2);
	DIR *dir;
	dir=opendir(argv[1]);
	ERROR_CHECK(dir,NULL,"opendir");
	struct dirent *p;
	while(p=readdir(dir))  //readdir返回dirent指针,若每读一次,会自动偏移
	{
		printf("%ld %d %d %s\n",p->d_ino,p->d_reclen,p->d_type,p->d_name);
	}
	closedir(dir);
	return 0;
}

ex2 : 以树形结构的形式输出指定目录下面的所有文件

#include <func.h>
int printDir(char* dirName,int width)
{
	DIR *dir;
	dir=opendir(dirName);
	ERROR_CHECK(dir,NULL,"opendir");
	struct dirent *p;
	char path[512]={0};
	while((p=readdir(dir)))
	{
		if(!strcmp(p->d_name,".")||!strcmp(p->d_name,".."))
		{
			continue;
		}
		printf("%*s%s\n",width,"-",p->d_name);
		sprintf(path,"%s%s%s",dirName,"/",p->d_name);
		if(4==p->d_type)
		{
			printDir(path,width+4);
		}
	}
	closedir(dir);
}

int main(int argc,char* argv[])
{
	ARGS_CHECK(argc,2);
	puts(argv[1]);
	printDir(argv[1],4);
	return 0;
}

ex: 用telldir返回目录流当前读取位置

#include <func.h>

int main(int argc,char* argv[])
{
	ARGS_CHECK(argc,2);
	DIR *dir;
	dir=opendir(argv[1]);
	ERROR_CHECK(dir,NULL,"opendir");
	struct dirent *p;
	off_t pos;
	while(p=readdir(dir))
	{
		printf("%ld %d %d %s\n",p->d_ino,p->d_reclen,p->d_type,p->d_name);
		if(!strcmp(p->d_name,"a.out"))
		{
			pos=telldir(dir);
		}
	}
	printf("--------------\n");
	seekdir(dir,pos);
	p=readdir(dir);
	printf("%s\n",p->d_name);
	closedir(dir);
	return 0;
}

练习:

1.传递任意一个目录路径,能够显示该目录的ls -l的效果

#include <func.h>

const char *romod[]={"---","--x","-w-","-wx","r--","r-x","rw-","rwx"};

void DetoBi(int n, int a[])
{
	int index = 0, i;
	do{
		i = n % 2;
		n = n / 2;
		a[index++] = i;
	}while(n);
}

int main(int argc, char* argv[])
{
	if(argc != 2)
	{
		printf("error args!\n");
		return -1;
	}
	struct stat buf;
	int ret = stat(argv[1], &buf);
	if(-1 == ret)
	{
		perror("stat");
	}
	printf("%ld ", buf.st_ino);

	int mod = buf.st_mode, a[16] = {0};
//	printf("%x",buf.st_mode);
	DetoBi(mod, a);
    if(a[15] == 1) printf("-");
	else printf("d");
	int res1 = 0, res2 = 0, res3 = 0;
	res1 = a[8] * 4 + a[7] * 2 + a[6] * 1;
	res2 = a[5] * 4 + a[4] * 2 + a[3] * 1;
	res3 = a[2] * 4 + a[1] * 2 + a[0] * 1;
	printf("%s%s%s ",romod[res1],romod[res2],romod[res3]);


	printf("%ld %d %d %ld ", buf.st_nlink, buf.st_uid, buf.st_gid, buf.st_size);


	char ori_date[100] = {0}; 
  	sprintf(ori_date,"%s",ctime(&buf.st_mtim));
	int i = 0, len = strlen(ori_date), num = 0;
	while(ori_date[i]!=' ') i++;
	i++;
	int xnum = 0;
	char date[100] = {0};
	for(;i<len;i++)
	{
		if(ori_date[i] == ':')
		{
			xnum++;   
			if(xnum == 2)
			{
				break;
			}
		}
		date[num++]=ori_date[i];
	}
	printf("%s %s\n", date, argv[1]);	

	return 0;
}

posted @ 2019-04-14 13:22  Mered1th  阅读(337)  评论(0编辑  收藏  举报