文件I/O

文件I/O是POSIX定义的一组函数,任何兼容POSIX标准的操作系统都支持文件I/O,不提供缓冲机制即每次读写操作都会引起系统调用

1、文件描述符

  • 内核利用文件描述符来访问文件。文件描述符在形式上是一个非负整数,实际上,它是一个索引值,指向内核为每一个进程所维护的该进程打开文件的记录表。
  • 当程序打开一个现有文件或者创建一个新文件时,内核向进程返回一个文件描述符。
  • 系统为每一个进程维护了一个文件描述符表,该表的值都是从0开始的,通常一个进程启动时,会打开 3 个文件:标准输入、标准输出和标准出错处理,对应的文件描述符分别为0(STDIN_FILENO),1(STDOUT_FILENO),2(STDERR_FILENO)。

2、文件I/O函数

2.1 open

头 文 件 :#include<fcntl.h>

函数原型:int open(const char *path, int oflag,  mode_t mode)

功    能   :打开一个文件,成功返回文件描述符,失败返回 EOF

注    意   :

  • 打开文件只使用前两个参数,创建文件时第三个参数指定新文件的权限;
  • open只能打开设备文件,而不能创建设备文件;

参    数   :

  • path           路径名(文件名)
  • oflag          标志位
oflag标志位的值
O_RDONLY   以只读的方式打开一个文件这三个参数互斥
O_WRONLY   以只写的方式打开一个文件
O_RDWR       以读写的方式打开一个文件
O_CREAT      如果文件不存在,自动创建一个新文件,并用第三个参数为其设置权限(如 0666)
O_EXCL        文件存在,报错。用来测试文件是否存在
O_APPEND   每次写时追加到文件尾
O_TRUNC      如果文件存在,先删除文件中的原有数据
O_NOCTTY    不将该设备分配作为进程的控制终端

 

2.2 close

头 文 件 :#include<unistd.h>

函数原型:int open(int fd)

功    能   :关闭文件,成功返回0,失败返回 EOF

注    意   :

  • 程序结束自动关闭所有打开的文件
  • 文件关闭后,文件描述符不再代表文件

参    数   :

  • fd         文件描述符

2.3 read

头 文 件 :#include<unistd.h>

函数原型:ssize_t read(int fd, void *buf, size_t count)

功    能   :读取文件,成功返回实际读取的字节数,失败返回 EOF

注    意   :

  • 读到文件末尾返回0

参    数   :

  • fd         文件描述符
  • buf       缓冲区首地址
  • count   读取的大小

2.4 write

头 文 件 :#include<unistd.h>

函数原型:ssize_t write(int fd, void *buf, size_t count)

功    能   :写文件,成功返回实际写入的字节数,失败返回 EOF

注    意   : count应该小于等于缓冲区buf 的大小

参    数   :

  • fd         文件描述符
  • buf       缓冲区首地址
  • count   写入的大小

2.4.1 例子:将键盘输入的内容写入文件,直到输入quit

#include<stdio.h>
#include<unistd.h> //write
#include<fcntl.h> //open
#include<string.h> //strcmp

int main(int argc, char *argv[]) 
{
    int fd;
    char buf[20];
    if(argc < 2)
    {
        printf("Usage: %s  <dst_file>",argv[0]);
        return -1;
    }
    if ((fd = open(argv[1], O_WRONLY|O_TRUNC|O_CREAT, 0666))<0)  
    {
        perror("open");
        return -1; 
    }
    while(fgets(buf, 20, stdin) != NULL) //从键盘接收字符串
    {
        if (strcmp("quit\n",buf) == 0) //输入quit退出
        {
            break;
        }
        
        write(fd, buf, strlen(buf));
    }
    close(fd);
    return 0;
}

2.4.2 例子:用文件IO复制文件

#include<stdio.h>
#include<fcntl.h>
#include<unistd.h>

int main(int argc, char *argv[]) 
{
    int fds,fdt,n;
    char buf[64];
    if(argc < 3)
    {
        printf("Usage: %s <src_file> <dst_file>",argv[0]);
        return -1;
    }
    if ((fds = open(argv[1], O_RDONLY))<0)  
    {
        perror("open src_file");
        return -1; 
    }
    if ((fdt = open(argv[2], O_WRONLY|O_TRUNC|O_CREAT, 0666))<0)  
    {
        perror("open dst_file");
        return -1; 
    }
    while ((n = read(fds, buf, 64)))
    {
        write(fdt, buf, n);
    }
    close(fds);
    close(fdt);
    return 0;
}

2.5 lseek

头 文 件 :#include<unistd.h>

函数原型:off_t lseek(int fd, off_t offset, intt whence)

功    能   :定位文件的读写位置,成功返回定位的位置相对于文件开始处的偏移量,失败返回 EOF

注    意   :

  •  offset+whence 不能小于0;
  • 标准I/O fseek成功返回0
  • 只能用在可定位(可随机访问) 文件操作中,管道、套接字和大部分字符设备文件是不可定位的

参    数   :

  • fd           文件描述符
  • offset     偏移量
  • whence  基准点  SEEK_SET 文件开头   SEEK_CUR 当前位置    SEEK_END 文件末尾

2.6 opendir

头 文 件 :#include<dirent.h>

函数原型:DIR *opendir(const char *name)

功    能   :打开目录文件,成功返回目录流指针,出错返回 NULL

注    意   :

  • DIR是用来描述打开的文件目录的结构体类型

参    数   :name 文件地址

2.7 readdir

头 文 件 :#include<dirent.h>

函数原型:struct dirent *readdir(DIR *dirp)

功    能   :读取目录流中的内容,成功返回目录流dirp中下一个目录项,出错返回 NULL

注    意   :

  • struct dirent 是用来描述目录流中一个目录项的结构体类型,包含成员char d_name[256]

参    数   :dirp 目录流指针

2.8 closedir

头 文 件 :#include<dirent.h>

函数原型:int closedir(DIR *dirp)

功    能   :关闭目录文件,成功返回0,出错返回EOF

参    数   :dirp 目录流指针

2.8.1 例子:打印指定文件目录下的所有文件名称

#include <stdio.h>
#include <dirent.h>
int main(int argc, char const *argv[])
{
    DIR *dirp; //文件流指针
    struct dirent *dirt; //文件结构体指针

    if (argc < 2)
    {
        printf("Usage: %s <dir>\n", argv[0]);
        return -1;
    }
    if ((dirp = opendir(argv[1])) == NULL)
    {
        perror("opendir");
        return -1;
    }
    while ((dirt = readdir(dirp)) != NULL)
    {
        printf("%s\n",dirt->d_name);
    }
    closedir(dirp);
    return 0;
}

2.9 chmod、fchmod

头 文 件 :#include<sys/stat.h>

函数原型:int chmod(const char *path, mode_t mode);

                      int fchmod(int fd, mode_t mode);

功    能   :修改文件访问权限,成功返回0,出错返回EOF

注    意   :使用这两个函数本身就要有权限,文件所有者或者root用户

参    数   :fd   文件描述符

2.9  stat/lstat/fstat

头 文 件 :#include<sys/stat.h>

函数原型:int stat(const char *path, struct stat *buf);

                      int lstat(const char *path, struct stat *buf); 

                      int fstat(int fd, struct stat *buf));

功    能   :获取文件属性,成功返回0,出错返回EOF

注    意   :

  • stat获取的是目标文件的属性如果path是一个符号链接文件,则stat获取的是符号链接文件指向的文件的属性
  • lstat获取的是链接文件的属性;

参    数   :fd   文件描述符

2.9.1 例子:打印文件相关内容:    类型 权限 大小 最后修改时间 文件名

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <time.h>

int main(int argc, char const *argv[])
{
    struct stat buf;
    struct tm *tp;

    if (argc < 2)
    {
        printf("Usage: %s <dst_file> \n", argv[0]);
        return -1;
    }
    if (lstat(argv[1], &buf) < 0)
    {
        perror("lstat");
        return -1;
    }
    switch (buf.st_mode & __S_IFMT) //判断文件类型 详见“man 7 inode”
    {
    case __S_IFDIR:
        printf("d");
        break;
    case __S_IFBLK:
        printf("b");
        break;
    default:
        printf("-");
        break;
    }
    for (int i = 8; i >= 0; i--)
    {
        if (buf.st_mode & (1 << i)) //buf.st_mode 低9位对应权限 rwxrwxrwx
        {
            switch (i % 3)
            {
            case 2:
                printf("r");
                break;
            case 1:
                printf("w");
                break;
            case 0:
                printf("x");
                break;
            }
        }
        else
        {
            printf("-");
        }
    }
    printf(" %lu", buf.st_size);
    tp = localtime(&buf.st_mtime);
    printf(" %d-%02d-%02d", tp->tm_year + 1900, tp->tm_mon + 1, tp->tm_mday);
    printf(" %s\n", argv[1]);
    return 0;
}

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值