Linux学习日记8——文件IO函数

学习视频链接:

黑马程序员-Linux系统编程_哔哩哔哩_bilibiliicon-default.png?t=N7T8https://www.bilibili.com/video/BV1KE411q7ee?p=52&spm_id_from=pageDriver

目录

一、系统调用(内核提供的函数)

二、open / close 函数

2.1 linux 下查看 open 命令

2.2 函数原型

2.3 操作

2.4 O_CREAT

2.5 O_TRUNC

2.6 常见错误

三、read / write 函数

3.1 函数原型

3.2 查看函数用法

3.3 实现拷贝操作

3.4 与 C 语言标准 IO 库比较

3.5 库函数与系统调用


一、系统调用(内核提供的函数)

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

只有系统调用才能进内核空间,才能调用驱动程序,驱动硬件工作。

二、open / close 函数

2.1 linux 下查看 open 命令

man 2 open

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

2.2 函数原型

int open(const char *pathname, int flags);

int open(const char *pathname, int flags, mode_t mode);

int close(int fd);

1、int open(const char *pathname, int flags);

pathname:路径名

flags:三个参数 O_RDONLY (以只读方式打开文件),O_WRONLY (以只写方式打开文件),O_RDWR (以可读写方式打开文件)

上述三种旗标是互斥的,也就是不可同时使用,但可与下列的标志利用 OR(|) 运算符组合

O_CREAT若欲打开的文件不存在则自动建立该文件
O_EXCL如果O_CREAT 也被设置,此指令会去检查文件是否存在。文件若不存在则建立该文件,否则将导致打开文件错误。此外,若O_CREAT与O_EXCL同时设置,并且欲打开的文件为符号连接,则会打开文件失败
O_NOCTTY如果欲打开的文件为终端机设备时,则不会将该终端机当成进程控制终端机
O_TRUNC若文件存在并且以可写的方式打开时,此旗标会令文件长度清为0,而原来存于该文件的资料也会消失
O_APPEND当读写文件时会从文件尾开始移动,也就是所写入的数据会以附加的方式加入到文件后面
O_NONBLOCK以不可阻断的方式打开文件,也就是无论有无数据读取或等待,都会立即返回进程之中
O_NDELAY同O_NONBLOCK
O_SYNC以同步的方式打开文件.
O_NOFOLLOW如果参数pathname所指的文件为一符号连接,则会令打开文件失败
O_DIRECTORY如果参数pathname所指的文件并非为一目录,则会令打开文件失败。 注:此为Linux2. 2 以后特有的标志,以避免一些系统安全问题

return:函数执行正确 返回新的文件描述符(整数),错误 返回 -1 并且 errno 被设置成相应的值

2、int open(const char *pathname, int flags, mode_t mode);

mode_t mode:8 进制整型,用来设置权限(权限大于当前用户 umask 时,会让文件权限变成 umask 对应的权限)

2.3 操作

新建一个 C 语言程序

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

 把光标移动到函数上,键入 2K 跳转到函数定义

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

 发现使用 open 需要 3 个头文件,其中前两个可以简写为一个头文件:unistd.h

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

测试运行

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

2.4 O_CREAT

删除 dict.txt,修改 C 程序源码

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

权限与设定的 644 是一样的 

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

2.5 O_TRUNC

往 dict.txt 里面添加一点东西

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

修改C语言文件 

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

 文件被清空了

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

2.6 常见错误

1、文件不存在错误

errno 记录着错误数字,把错误数字传入到 strerror 中返回错误信息

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

查看更多的 strerror 用法可以使用 linux 命令 man(strerror)

2、以写方式打开只读文件

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

 运行结果:

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

3、以只写文件打开目录

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

运行结果: 

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

三、read / write 函数

3.1 函数原型

ssize_ t read(int fd, void *buf, size_t count);

ssize_t write(int fd, const void *buf, size_t count);

3.2 查看函数用法

1、man 2 read

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

fd:文件描述符

buf:保存数据的缓冲区

count:缓冲区的大小

返回值:成功返回实际读到的字节数,读到文末返回 0,失败返回 -1 并且 errno 被设置成相应的值

2、man 2 write

ssize_t write(int fd, const void *buf, size_t count);

fd:文件描述符

buf:待写出数据的缓冲区

count:实际要写出的内容的大小

返回值:成功返回写入的字节数,失败赶回 -1 并且 errno 被设置成相应的值

3.3 实现拷贝操作

1、C 语言代码

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

运行结果:

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

2、添加错误信息

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

3.4 与 C 语言标准 IO 库比较

1、执行原理

C语言标准库函数 fputc 函数调用了系统函数 write 函数执行写操作

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

2、使用标准库函数和 read write 读写速度对比

(1) 准备这三个文件

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

2、编译执行两个C语言文件,比较他们的拷贝速度

发现使用标准 IO 库函数执行拷贝命令稍微快一点

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

3、查看这两个函数执行时候调用了多少次 read 和 write

strace ./a_cp_b

发现执行了很多次 read 和 write ,每次读写 4096 个字节

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

执行用系统调用函数编写的 C 程序

每次读写5个字节,所以更慢

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

  

4、结果分析

我们调用系统函数,系统直接就执行对应的指令,调用 C语言标准 IO库的时候,如下图:

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAaGVyYi5kcg==,size_20,color_FFFFFF,t_70,g_se,x_16

 C语言库中有用户级别的缓冲,虽然表现在程序上的是一次读取一个字节,但是在 fputc 函数中有一个缓冲区,当缓冲区达到 4096 个字节的时候才会调用系统函数执行 read 和 write,这个被称为预读入缓输出。

5、结论

即使学了系统调用,能使用库函数的时候尽量使用库函数。有的时候不想使用缓输出的机制,可以使用系统调用。 

3.5 库函数与系统调用

未使用系统调用的库函数,其执行效率通常要比系统调用的高。因为使用系统调用时,需要上下文的切换及状态转换(用户态转向核心态)

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

herb.dr

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

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

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

打赏作者

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

抵扣说明:

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

余额充值