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;
}

1600805-20190414132141604-850154286.png

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;
}

1600805-20190414132121120-952259588.png

转载于:https://www.cnblogs.com/Mered1th/p/10704794.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值