linux文件、目录操作(用户态)

1、文件操作
fopen:

从大纲获取文件头文件,和函数原型

path:文件路径

mode:

r:只读,ptr位于文件开头

r+:读写,ptr位于文件开头,文件不存在不创建文件

w:只写,ptr位于文件开头,文件不存在创建文件,文件存在截断文件为0

w+:读写:ptr位于文件开头,文件存在创建文件,文件存在截断文件为0

a:追加,ptr位于文件末尾,文件不存在创建文件

a+:读和追加,打开时ptr位于文件开头,写入时跳到文件末尾开始

fread:

从stream读取nmemb个size大小的数据,写入ptr指向的内存

返回值:成功返回read到的字节数/size,失败或者读到文件末尾返回0

fwrite:

从str指向内存中读取nmemb个大小为size的数据,写入stream文件流中

返回值:成功返回write的字节数/size,失败或者读到内存返回0

#include <func.h>
int main(int argc, char *argv[]){
    ARGS_CHECK(argc, 2);
    FILE* file = fopen(argv[1], "a+");//以追加方式打开文件
    ERROR_CHECK(file, NULL, "fopen");
    printf("before write ptr = %ld\n", ftell(file));
    char buf[10] = {0};
    fwrite(buf, 1, 9, file);
    printf("after write ptr = %ld\n", ftell(file));
    int ret = fclose(file);
    ERROR_CHECK(ret, EOF, "fclose");
}
chmod:

返回值:

成功0,失败-1

#include <func.h>
int main(int argc, char *argv[]){
    ARGS_CHECK(argc, 2);
    int ret = chmod(argv[1], 0777);
    ERROR_CHECK(ret, -1, "fclose");
}
2、目录操作

目录存储原理:在linux中,目录是一种特别的文件,总体大小固定,数据块中把很多文件的文件名和索引节点存放在一起。由于文件名大小不一,为了避免磁盘碎片和支持频繁增加修改,所以以链式结构来存储各种文件信息,节点是dirent节点,要访问节点实际以来d_off属性

2.1、获取当前目录

将当前工作目录绝对路径放到buf中,若buf空间不够返回NULL,并设置erron

#include <func.h>
int main(){
    char buf[1024] = {0};
    char* ret = getcwd(buf, sizeof(buf));
    ERROR_CHECK(ret, NULL, "getcwd");
    puts(buf);
}
2.2、改变当前目录

更改进程当前工作目录为buf

返回值:失败返回-1

#include <func.h>
int main(int argc, char *argv[]){
    int ret = chdir(argv[1]);
    ERROR_CHECK(ret, -1, "chdir");
    char buf[1024] = {0};
    char* ret2 = getcwd(buf, sizeof(buf));
    ERROR_CHECK(ret2, NULL, "getcwd");
    puts(buf);
}
2.3、创建和删除目录:

使用path创建一个模式为mode的新目录

#include <func.h>
int main(int argc, char *argv[]){
    int ret = mkdir(argv[1], 0777);
    ERROR_CHECK(ret, -1, "mkdir");
}

 

删除path所指目录

#include <func.h>
int main(int argc, char *argv[]){
    int ret = rmdir(argv[1]);
    ERROR_CHECK(ret, -1, "rmdir");
}
2.4、opendir

打开一个和name相应的目录流,并返回指向这个目录流的一个指针,这个目录流指向这个目录的第一个项。

2.5、closedir

和打开关闭文件一样,打开目录流之后要关闭它

2.6、readdir

返回值:

发生错误,返回NULL,并设置erron,成功返回一个指向dirent(dirp的下一个)的指针,到达目录流末尾,返回NULL

2.7、其他目录流相关函数:

1、rewinddir

重定位到目录流头部

2、seekdir

用来设置目录流当前读取位置

3、telldir

返回目录流当前读取位置

 2.8、应用:ls实现、tree实现

//ls
#include <func.h>
int main(int argc, char *argv[]){
    ARGS_CHECK(argc, 2);
    DIR* dirp = opendir(argv[1]);
    ERROR_CHECK(dirp, NULL, "opendir");
    struct dirent *pdirent = NULL;
    struct stat statbuf;//获取文件属性
    int ret = chdir(argv[1]);
    ERROR_CHECK(ret, -1, "chdir");
    while((pdirent = readdir(dirp)) != NULL){
        ret = stat(pdirent->d_name, &statbuf);
        ERROR_CHECK(ret, -1, "stat");
        printf("%6o %ld %s %s %8ld %s %s\n",
        statbuf.st_mode,//u、g、o的rwx等等
        statbuf.st_nlink,//硬链接数
        getpwuid(statbuf.st_uid)->pw_name,//通过uid获取用户名
        getgrgid(statbuf.st_gid)->gr_name,//获取组id
        statbuf.st_size,//总大小,bytes
        ctime(&statbuf.st_mtime),//ctime参数是从某一时刻开始到现在的秒数
        pdirent->d_name);//目录名
    }
    closedir(dirp);
}
//tree
#include <func.h>
void DfsPrint(char*, int);
int main(int argc, char *argv[]){
    ARGS_CHECK(argc, 2);
    puts(argv[1]);
    DfsPrint(argv[1], 4);
    return 0;
}
void DfsPrint(char *path, int width){
    DIR *dirp = opendir(path);
    ERROR_CHECK(dirp, NULL, "opendir");
    struct dirent *pdirent = NULL;
    struct stat statbuf;
    char newPath[1024] = {0};
    while((pdirent = readdir(dirp)) != NULL){
        if(strcmp(pdirent->d_name, ".") == 0 || strcmp(pdirent->d_name, "..") == 0){
            continue;
        }
        printf("├");
        for(int i = 0; i < width; ++i){
            printf("-");
        }
        puts(pdirent->d_name);
        if(pdirent->d_type == DT_DIR){
            sprintf(newPath, "%s/%s", path, pdirent->d_name);
            DfsPrint(newPath, width + 4);
        }
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值