学习视频链接:
黑马程序员-Linux系统编程_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV1KE411q7ee?p=52&spm_id_from=pageDriver
目录
一、系统调用(内核提供的函数)
只有系统调用才能进内核空间,才能调用驱动程序,驱动硬件工作。
二、open / close 函数
2.1 linux 下查看 open 命令
man 2 open
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 语言程序
把光标移动到函数上,键入 2K 跳转到函数定义
发现使用 open 需要 3 个头文件,其中前两个可以简写为一个头文件:unistd.h
测试运行
2.4 O_CREAT
删除 dict.txt,修改 C 程序源码
权限与设定的 644 是一样的
2.5 O_TRUNC
往 dict.txt 里面添加一点东西
修改C语言文件
文件被清空了
2.6 常见错误
1、文件不存在错误
errno 记录着错误数字,把错误数字传入到 strerror 中返回错误信息
查看更多的 strerror 用法可以使用 linux 命令 man(strerror)
2、以写方式打开只读文件
运行结果:
3、以只写文件打开目录
运行结果:
三、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
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 语言代码
运行结果:
2、添加错误信息
3.4 与 C 语言标准 IO 库比较
1、执行原理
C语言标准库函数 fputc 函数调用了系统函数 write 函数执行写操作
2、使用标准库函数和 read write 读写速度对比
(1) 准备这三个文件
2、编译执行两个C语言文件,比较他们的拷贝速度
发现使用标准 IO 库函数执行拷贝命令稍微快一点
3、查看这两个函数执行时候调用了多少次 read 和 write
strace ./a_cp_b
发现执行了很多次 read 和 write ,每次读写 4096 个字节
执行用系统调用函数编写的 C 程序
每次读写5个字节,所以更慢
4、结果分析
我们调用系统函数,系统直接就执行对应的指令,调用 C语言标准 IO库的时候,如下图:
C语言库中有用户级别的缓冲,虽然表现在程序上的是一次读取一个字节,但是在 fputc 函数中有一个缓冲区,当缓冲区达到 4096 个字节的时候才会调用系统函数执行 read 和 write,这个被称为预读入缓输出。
5、结论
即使学了系统调用,能使用库函数的时候尽量使用库函数。有的时候不想使用缓输出的机制,可以使用系统调用。
3.5 库函数与系统调用
未使用系统调用的库函数,其执行效率通常要比系统调用的高。因为使用系统调用时,需要上下文的切换及状态转换(用户态转向核心态)