文件IO简介(1)

标准C库IO函数

FILE *fopen(const char *filename, const char *mode)
  • 返回值(FILE * fp):文件描述符(文件指针),索引到对应的磁盘文件
  • 文件读写指针位置:在文件内部有一个位置指针,用来指向当前读写到的位置,也就是读写到第几个字节。在文件打开时,该指针总是指向文件的第一个字节。

注意:这个文件内部的位置指针与C语言中的指针不是一回事。位置指针仅仅是一个标志,表示文件读写到的位置,也就是读写到第几个字节,它不表示地址。文件每读写一次,位置指针就会移动一次,它不需要你在程序中定义和赋值,而是由系统自动设置,对用户是隐藏的。

  • I/O缓冲区(内存地址):通过寻址找到对应的内存块并将数据从内存刷新到磁盘

刷新方法:

  • 刷新缓冲区:fflush
  • 缓冲区已满(默认Buffer 8192byte)
  • 正常关闭文件:a.fclose b.return(main函数) c.exit(main函数)

  •  直接调用标准C库的I/O会有缓冲区存在,在系统内部传递文件时效率高(可以减少系统I/O执行次数)
  • 在网络传输时选用系统I/O可以提高网络通信效率(及时发送文件)

文件打开、创建、关闭

  • 头文件
 #include <sys/types.h>
 #include <sys/stat.h>
 #include <fcntl.h>
  • 函数
int open(const char *pathname, int flags);
  • 作用:打开一个已经存在的文件
  • 参数
    • pathname:要打开的文件路径(必选项)
    • flags:对文件的操作权限设置还有其他的设置
      • 必选项:O_RDONLY, O_WRONLY, O_RDWR  这三个权限是互斥的
      •  可选项:O_CREAT 文件不存在,创建新文件
  • 返回值:返回一个新的文件描述符,如果调用失败,返回-1
  • 函数
int open(const char *pathname, int flags, mode_t mode);
  • 参数
    •   mode:八进制的数,表示创建出的新的文件的操作权限,比如:0775
    • 最终的权限:mode & ~umask(umask:抹去某些权限)

  • 函数

close(fd);
  • 作用:关闭文件
  • 函数
#include <stdio.h>
void perror(const char *s);    
  • 作用:打印errno对应的错误描述
    • errno:属于Linux系统函数库,库里面是一个全局变量,记录的是最近的错误号
  • 参数:用户描述,比如hello,最终输出的内容是hello:xxx(实际的错误描述)
int fd = open("a.txt", O_RDWR | O_CREAT, 0777);
if(fd == -1){
    perror("open");
}

作用:创建并打开一个文件,并赋给其相应的权限,如果失败就返回对应的错误描述

文件读取、写入操作

  • 头文件
#include <unistd.h>
  • 函数
ssize_t read(int fd, void *buf, size_t count);
  • 参数
    • buf:缓冲区,需要读取数据存放的地方,数组的地址(传出参数)

      • 注:传入参数为本身有值,传入函数让函数使用,传出参数是本身没值,传入函数让函数赋值。

    • count:指定的数组的大小

  • 返回值
    • 成功:
      • >0:返回实际的读取到的字节数

      • -0:文件已经读取完了

  • 函数

ssize_t write(int fd, const void *buf, size_t count);
  • 返回值
    • 成功:实际写入的字节数
    • 失败:返回-1,并设置errno

文件指针移动函数

  • 头文件
#include <sys/types.h>
#include <unistd.h>
  • 函数
off_t lseek(int fd, off_t offset, int whence);
  • 参数
    • offset:偏移量

    • whence:

      • SEEK_SET:设置文件指针的偏移量

      • SEEK_CUR:设置偏移量:当前位置 + 第二个参数offset的值

      • SEEK_END:设置偏移量:文件大小 + 第二个参数offset的值

  • 返回值:返回文件指针的位置,错误返回-1
  • 作用
    • 移动文件指针到头文件:lseek(fd, 0, SEEEK_SET);

    • 获取当前文件指针的位置:lseek(fd, 0, SEEK_CUR);

    • 获取文件长度:lseek(fd, 0, SEEK_END);

    • 拓展文件的长度,当前文件10b,110b增加了100个字节:lseek(fd, 100, SEEK_END);

 查看文件信息函数

  • 头文件
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
  • 函数
int stat(const char *pathname, struct stat *statbuf);
  • 作用:获取一个文件相关的信息
  • 参数
    • pathname:文件路径

    • statbuf:结构体变量,传出参数,用于保存获取到的文件信息

  • 返回值
    • 成功:返回0

    • 失败:返回-1,设置errno

  • 函数
int lstat(const char *pathname, struct stat *statbuf);
  • 作用:获取软连接文件的信息(如果用stat输出软连接文件的信息会输出所指向的文件的信息)

模拟实现ls -l命令

#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdio.h>
#include <pwd.h>
#include <grp.h>
#include <time.h>
#include <string.h>


int main(int argc, char *argv[]){
    //判断输入的参数是否正确
    if(argc < 2){
        printf("%s filename\n", argv[0]);
        return -1;
    }

    //通过stat函数获取用户传入的文件的信息
    struct stat st;
    int ret = stat(argv[1], &st);
    if(ret == -1){
        perror("stat");
        return -1;
    }

    //获取文件类型和文件权限
    char perms[11] = {0};//用于保存文件类型和文件权限的字符串
    switch(st.st_mode & S_IFMT){
        case S_IFLNK:
            perms[0] = 'l';
            break;
        case S_IFDIR:
            perms[0] = 'd';
            break;
        case S_IFREG:
            perms[0] = '-';
            break;
        case S_IFBLK:
            perms[0] = 'b';
            break;
        case S_IFCHR:
            perms[0] = 'c';
            break;
        case S_IFSOCK:
            perms[0] = 's';
            break;
        case S_IFIFO:
            perms[0] = 'p';
            break;
        default:
            perms[0] = '?';
            break;
    }

    //判断文件的访问权限
    //文件所有者
    perms[1] = (st.st_mode & S_IRUSR) ? 'r' : '-';
    perms[2] = (st.st_mode & S_IWUSR) ? 'w' : '-';
    perms[3] = (st.st_mode & S_IXUSR) ? 'x' : '-';

    //文件所在组
    perms[4] = (st.st_mode & S_IRGRP) ? 'r' : '-';
    perms[5] = (st.st_mode & S_IWGRP) ? 'w' : '-';
    perms[6] = (st.st_mode & S_IXGRP) ? 'x' : '-';

    //其他人
    perms[7] = (st.st_mode & S_IROTH) ? 'r' : '-';
    perms[8] = (st.st_mode & S_IWOTH) ? 'w' : '-';
    perms[9] = (st.st_mode & S_IXOTH) ? 'x' : '-';

    //硬链接数
    int linkNum = st.st_nlink;

    //文件所有者
    char * fileUser = getpwuid(st.st_uid)->pw_name;

    //文件所在组
    char * fileGrp = getgrgid(st.st_gid)->gr_name;

    //文件大小
    long int fileSize = st.st_size;

    //获取修改的时间
    char * time = ctime(&st.st_mtime);

    char mtime[512] = {0};
    //拷贝函数
    strncpy(mtime, time, strlen(time) - 1);

    char buf[1024];
    sprintf(buf, "%s %d %s %s %ld %s %s", perms, linkNum, fileUser, fileGrp, fileSize, mtime, argv[1]);

    printf("%s %d %s %s %ld %s %s\n", perms, linkNum, fileUser, fileGrp, fileSize, mtime, argv[1]);

    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值