【C语言】文件IO

目录

1. 什么是文件IO

1.1概念

1.2特点

1.3操作

2. 函数接口

2.1打开文件open()

2.2关闭文件

2.3读写文件

2.3.1读文件

2.3.2写文件

2.4文件的定位操作

标准IO和文件IO总结


 

1. 什么文件IO

1.1概念

又称系统IO,是系统调用,是操作系统提供的接口函数。

posix中定义的一组用于输入输出的函数

POSIX接口 (英语:Portable Operating System Interface)可移植操作系统接口

1.2特点

(1) 没有缓存机制,每次调用都会引起系统调用。

(2) 围绕文件描述符进行操作,非负整数(>=0),依次分配

(3) 文件IO默认打开三个文件描述符,分别是0(标准输入),1(标准输出),2(标准错误)

(4) 操作任意类型的文件bc -lsp(除了d目录文件)

1.3操作

打开文件:open

关闭文件:close

读写操作:read、write

定位操作:lseek

2. 函数接口

2.1打开文件open()

int open(const char *pathname, int flags);
功能:打开文件
参数:pathname:文件路径名
     flags:打开文件的方式
            O_RDONLY:只读
            O_WRONLY:只写
            O_RDWR:可读可写
            O_CREAT:不存在创建
            O_TRUNC:存在清空
            O_APPEND:追加   
返回值:成功:文件描述符
       失败:-1

                当第二个参数中有O_CREAT选项时,需要给open函数传递第三个参数,指定创建文件的权限 

int open(const char *pathname, int flags, mode_t mode);
最后权限=创建出来的文件指定权限值&(~umask)  
    例如:指定权限为0666(8进制)
最终权限为:0666 & (~umask) = 0666&0775 = 0664

例子:

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

int main(int argc, char const *argv[])
{
    int fd;
    //fd = open("a.c",O_WRONLY|O_CREAT|O_TRUNC,0666);
    fd = open("b.c",O_RDONLY);
    if(fd < 0)
    {
        perror("open err");
        return -1;
    }
    printf("fd:%d\n",fd);
    return 0;
}

思考:文件IO和标准IO的打开方式的对应关系.

标准IO

文件IO

r

O_RDONLY只读

r+

O_RDWR可读可写

w

O_WRONLY|O_CREAT|O_TRUNC,0666 只写,不存在创建,存在清空

w+

O_RDWR|O_CREAT|O_TRUNC,0666可读可写,不存在创建,存在清空

a

O_WRONLY|O_CREAT|O_APPEND,0666

只写,不存在创建,存在追加

a+

O_RDWR|O_CREAT|O_APPEND,0666

可读可写,不能存在创建,存在追加

注意:O_CREAT需要加权限

2.2关闭文件

int close(int fd);
功能:关闭文件
参数:fd:文件描述符

2.3读写文件

2.3.1读文件

ssize_t read(int fd, void *buf, size_t count);
功能:从一个已打开的可读文件中读取数据
参数: fd  文件描述符
      buf  存放位置
      count  期望的个数
返回值:成功:实际读到的个数(小于期望值说明实际没这么多)
       返回0:表示读到文件结尾
       返回-1:表示出错,并设置errno号

fgetc->EOF 失败或末尾

fgets->NULL失败或末尾

fread->0失败或末尾

read->-1失败 0末尾

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

int main(int argc, char const *argv[])
{
    int fd;
    char buf[32]="";

    fd = open("a.c", O_RDONLY);
    if (fd < 0)
    {
        perror("open err");
        return -1;
    }
    printf("fd:%d\n", fd);
    read(fd,buf,10);    //向fd文件中读取10个字符
    printf("buf: %s\n",buf);

    close(fd);
    return 0;
}

2.3.2文件

ssize_t write(int fd, const void *buf, size_t count);
功能:向指定文件描述符中,写入 count个字节的数据。
参数:fd   文件描述符
      buf   要写的内容
      count  期望写入字节数
返回值:成功:实际写入数据的个数
              失败  : -1

例子:

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

int main(int argc, char const *argv[])
{
    int fd;
    char buf[32]="hello";

    fd = open("a.c", O_RDWR|O_CREAT|O_TRUNC,0666);
    if (fd < 0)
    {
        perror("open err");
        return -1;
    }
    printf("fd:%d\n", fd);

    write(fd,buf,5);
    write(fd,"haha",10);

    close(fd);
    return 0;
}

练习:文件IO实现cp功能。cp 源文件 新文件名

./a.iout src dest

思路:打开文件,循环读源文件内容,写到目标文件,当读到文件末尾循环结束,关闭两个文件。

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

int main(int argc, char const *argv[])
{
    int fd_src, fd_dest;
    char buf[32];
    ssize_t n;
    if (argc != 3)
    {
        printf("Usage: %s <srcfile> <destfile>\n", argv[0]);
        return -1;
    }

    //可读方式打开源文件
    fd_src = open(argv[1], O_RDONLY);
    if (fd_src < 0)
    {
        perror("open src file err");
        return -1;
    }

    //可写方式打开目标文件
    fd_dest = open(argv[2], O_WRONLY | O_CREAT | O_TRUNC, 0666);
    if (fd_dest < 0)
    {
        perror("open dest file err");
        return -1;
    }

    //循环读,只要读到内容就写入目标文件
    while ((n = read(fd_src, buf, 32)) > 0)
        write(fd_dest, buf, n);

    close(fd_src);
    close(fd_dest);
    return 0;
}

2.4文件定位操作

off_t lseek(int fd, off_t offset, int whence);
功能:设定文件的偏移位置
参数:fd:文件描述符
    offset: 偏移量  
        正数:向文件结尾位置移动
        负数:向文件开始位置
    whence: 相对位置
        SEEK_SET   开始位置
        SEEK_CUR   当前位置
        SEEK_END   结尾位置
补充:和fseek一样其中SEEK_SET,SEEK_CUR和SEEK_END和依次为0,1和2.

        返回值:成功:文件的当前位置
        失败:-1
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

int main(int argc, char const *argv[])
{
    int fd;
    char buf[32] = "";

    fd = open("a.c", O_RDWR | O_CREAT | O_TRUNC, 0666);
    if (fd < 0)
    {
        perror("open err");
        return -1;
    }
    printf("fd:%d\n", fd);

    // write(fd, "66666", 10);
    // lseek(fd,0,SEEK_SET);  //定位到文件开头
    // read(fd, buf, 10);
    // printf("%s\n", buf);

    lseek(fd,10,2);            //定位到末尾后10个字符单位
    write(fd,"k",1);

    off_t off = lseek(fd,0,SEEK_END);    //利用结尾偏移求出当前长度
    printf("%ld\n",off);

    close(fd);
    return 0;
}

练习:向文件中第 10 位置后面写一个字符,在文件此时的位置,后第 20个位置处,写一行字符串hello进去,求此时文件的长度。

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

int main(int argc, char const *argv[])
{
    int fd;
    char buf[32] = "";

    fd = open("a.c", O_RDWR | O_CREAT | O_TRUNC, 0666);
    if (fd < 0)
    {
        perror("open err");
        return -1;
    }
    printf("fd:%d\n", fd);

    lseek(fd,10,SEEK_SET);
    write(fd,"o",1);

    lseek(fd,20,SEEK_CUR);
    write(fd,"hello",5);

    off_t off = lseek(fd,0,SEEK_END);
    printf("len=%ld\n",off);

    close(fd);
    return 0;
}

标准IO和文件IO总结

标准IO

文件IO

概念

C库中定义的一组用于输入输出的函数

posix中定义的一组用于输入输出的函数

特点

1. 有缓冲机制

2. 围绕流进行操作,FILE*

3. 默认打开三个流:stdin/stdout/stderr

4. 只能操作普通文件

5. 程序的可移植性更强

1. 无缓冲机制

2. 围绕着文件描述符,非负整数

3. 默认打开三个文件描述符:0/1/2

4. 可以操作除了目录以外的任意类型文件

函数

打开文件:fopen、freopen

关闭文件:fclose

读文件:fgetc/fgets/fread

写文件:fputc/fputs/fwrite

定位操作:rewind/fseek/ftell

打开文件:open

关闭文件:close

读文件: read

写文件:write

定位操作:lseek

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

WeeJot

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值