当前位置: 首页 > news >正文

Linux操作系统3-文件与IO操作1(从C语言IO操作到系统调用)

上篇文章:Linux操作系统2-进程控制3(进程替换,exec相关函数和系统调用)_execv系统调用-CSDN博客

本篇代码Gitee仓库:myLerningCode · 橘子真甜/linux学习 - 码云 - 开源中国 (gitee.com)

本篇重点:C语言基础IO与系统调用

目录

一. 文件相关知识

二. C语言的文件操作 

2.1 fopen

2.2 fclose 

2.3 fread

2.4 fwrite

2.5 fprintf

2.6 举例代码

三. 文件相关的IO系统调用

3.1 open

3.2 close 

3.3 write

3.4 read

3.5 举例代码操作 

四. OS是如何管理被打开的文件?

4.1 文件fd

 五. 下篇重点: 文件fd, Linux下一切皆文件


一. 文件相关知识

        在基础指令这篇文章 Linux基础1-基本指令2(你真的了解文件吗?)-CSDN博客 中,我们提到了文件的相关命令。总结一下

1 一个空文件也需要占用空间

2 文件 = 文件内容 + 文件属性

3 文件操作 = 操作文件内容 + 操作文件属性

4 我们使用文件路径+文件名标记一个文件

5 进程想要访问一个文件必须要先通过OS打开这个文件

        C语言为用户提供了文件操作,C++也有相关的文件操作。像这些语言级别的库函数提供的文件操作,都是对OS提供的文件操作系统调用的封装。所以,学习系统调用提供的文件操作有利于我们掌握语言级的文件操作

二. C语言的文件操作 

        C语言中的库函数为我们提供了很多操作文件的函数:fopen,fclose,fwrite,fread,fprintf,fscanf...等。

2.1 fopen

        fopen用于打开一个文件,其函数原型如下:

//所需头文件
#include <stdio.h>//函数原型
FILE* fopen(const char* filename, const char* mode);//filename,打开文件的路径。直接写名字默认在当前路径下查找//mode,打开的方式
"r" 只读方式打开
"w" 只写方式打开,默认会清空文件中的内容
"a" 只写,写的方式是追加
"b" 以二进制方式打开,一般配合r和w使用
"W+" 读写,没有文件会创建,写方式是清空文件从头开始写
"r+" 读写,从头开始写文件
"a+" 读写,追加写

2.2 fclose 

        用于关闭一个打开的文件

//函数原型
int fclose(FILE* stream);//关闭stream这个文件流(被fopen打开的文件流)

2.3 fread

        用于读一个文件中的数据

//函数原型
size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);ptr: 读取的文件存放在内存中的位置
size:读取文件中元素大小(以字节为单位)
nmemb:读取文件元素的数量
stream:读取文件的文件流(你要读取的文件)

2.4 fwrite

size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream);//参数和fread类似//只是功能是将ptr写入到stream这个文件流中

2.5 fprintf

//函数原型
int fprintf(FILE *stream, const char *format, ...);//用法和printf一样,不过是将数据写入到stream这个文件流中

其他文件操作都和上述文件操作类似。具体内容可以查找man手册

2.6 举例代码

        用一段代码来举例这些操作的用法

test.c

#include <stdio.h>
#include <string.h>
#include <stdlib.h>#include <unistd.h>
#define MY_FILENAME "log.txt"int main()
{// 1.写入数据FILE *fp = fopen(MY_FILENAME, "w");if (fp == NULL){// 打开文件失败perror("fopen");}// 2.使用fwrite写数据,写入三行 Hello worldconst char *buffer = "Hello World!\n";fwrite(buffer, sizeof(char), strlen(buffer), fp);fwrite(buffer, sizeof(char), strlen(buffer), fp);fwrite(buffer, sizeof(char), strlen(buffer), fp);// 3. 关闭文件fclose(fp);fp = NULL;return 0;
}

Makefile

test:test.cgcc -o $@ $^ -std=c99.PHONY:clean
clean:rm -rf test log.txt

 测试结果如下:

修改 log.txt 和 test.c 进行读取数据 

log.txt

Hello World!
Hello World!
Hello World!
YZC yzc 
abc 123
156 1sg 45qe1r 5h@#@ ^% 56 @# ^re8 5qh qer56h 16 32`7 tr314yt 9bm    v891-3 

test.c

#include <stdio.h>
#include <string.h>
#include <stdlib.h>#include <unistd.h>
#define MY_FILENAME "log.txt"int main()
{// 1.读取数据FILE *fp = fopen(MY_FILENAME, "r");if (fp == NULL){// 打开文件失败perror("fopen");}// 2.使用fread写数据,写入三行 Hello worldchar buffer[200];fread(buffer, sizeof(char), 200, fp);buffer[strlen(buffer) - 1] = '\0';  //将最后的'\n'变为'\0'printf("%s\n", buffer);// 3. 关闭文件fclose(fp);fp = NULL;return 0;
}

 测试结果:

      注意C语言的字符串默认在结尾有一个'0',而文本文件中末尾并没有'\0'。所以我们使用C语言接口读取文件后,如果是字符串,需要在末尾加上'\0' 

三. 文件相关的IO系统调用

3.1 open

        打开文件的系统调用

//所需头文件
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>//函数原型
int open(const char *pathname, int flags)                 //用于打开已经创建的文件
int open(const char *pathname, int flags, mode_t mode)    //用于打开和创建文件    //pathname 打开文件的名字
//flags    打开文件的方式
//mode     创建文件时候文件的权限//常见的flag
O_RDONLY 表示只读
O_WRONLY 只写
O_WRONLY 读写
O_APPEND 追加写
O_CREAT  没有这个文件要创建文件
O_TRUNC  打开文件的时候清空文件内容

3.2 close 

        关闭文件fd的系统调用

//文件关闭
#include <unistd.h>//函数原型
int close(int fildes);

3.3 write

        向文件写入数据

//头文件
#include <unistd.h>//函数原型
ssize_t write(int fd, const void *buf, size_t count);//fd 写入的文件fd//buf 要写的数据缓冲区来源//count 写入的字节个数//返回值,成功写入,返回写入的字符数,失败返回-1

buf是void* 的原因:在系统看来,任何数据都是二进制

3.4 read

        从文件中读取数据

//头文件
#include <unistd.h>//函数原型
ssize_t read(int fd, void *buf, size_t count)//将fd文件中的count字节数量的数据读取到buf中//返回0表示读取到文件结尾

3.5 举例代码操作 

 

#include <stdio.h>
#include <string.h>
#include <stdlib.h>#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#define MY_FILENAME "log.txt"int main()
{// 1.打开文件,方式是读写,没有文件创建,清空文件从头开始写。创建的文件权限是0666umask(0); // 清空系统的umaskint fd1 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_TRUNC, 0666);// 如果写入失败if (fd1 < 0){perror("open");return -1;}// 2.写入数据char buffer[64];int cnt = 5;while (cnt){sprintf(buffer, "YZC Hello World [%d]\n", cnt--); // 将数据写入缓冲区write(fd1, buffer, strlen(buffer));             // 向文件写入数据不需要添加'\0'}// 3.关闭文件描述符fdclose(fd1);// 4.读取这些数据int fd2 = open(MY_FILENAME, O_RDONLY);// 如果文件打开错误if (fd2 < 0){perror("open");return -1;}// 读取文件的时候,buffer最多读取sizeof(buf)个数据,由于有'\0'。所以要-1char *buf[64];ssize_t num = read(fd2, buf, sizeof(buf) - 1);printf("%s", buf);// 关闭文件close(fd2);return 0;
}

测试结果:

 语言级别的IO操作库函数都是对系统调用IO操作的封装

四. OS是如何管理被打开的文件?

        我们知道,OS通过PCB来管理进程。在OS中有很多进程,这些进程也会打开很多的文件。那么OS是如何管理这些被打开的文件的?

        OS为了管理被打开的文件,创建对应的内核数据结构 struct_file。这个结构体包含了文件的大量属性。

4.1 文件fd

         文件fd是什么东西?我们打印出来看看

#include <stdio.h>
#include <string.h>
#include <stdlib.h>#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#define MY_FILENAME "log.txt"int main()
{umask(0); //清楚umask码,仅仅修改该进程创建的文件int fd1 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_APPEND, 0666);int fd2 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_APPEND, 0666);int fd3 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_APPEND, 0666);int fd4 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_APPEND, 0666);int fd5 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_APPEND, 0666);printf("fd1:%d\n",fd1);  printf("fd2:%d\n",fd2);  printf("fd3:%d\n",fd3);  printf("fd4:%d\n",fd4);  printf("fd5:%d\n",fd5);  close(fd1);close(fd2);close(fd3);close(fd4);close(fd5);return 0;
}

fd为什么从3开始?

        因为C语言会默认打开三个输入输出流,stdin, stdout, stderr。

        即标准输入,标准输出,标准错误。它们占用了0 1 2 

通过stdin这个文件的结构体中的 _fileno 即可获取fd

#include <stdio.h>
#include <string.h>
#include <stdlib.h>#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#define MY_FILENAME "log.txt"int main()
{umask(0); //清楚umask码,仅仅修改该进程创建的文件int fd1 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_APPEND, 0666);int fd2 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_APPEND, 0666);int fd3 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_APPEND, 0666);int fd4 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_APPEND, 0666);int fd5 = open(MY_FILENAME, O_WRONLY | O_CREAT | O_APPEND, 0666);printf("stdin->fd [%d]\n",stdin->_fileno);  printf("stdout->fd [%d]\n",stdout->_fileno);  printf("stderr->fd [%d]\n",stderr->_fileno);  printf("fd1:%d\n",fd1);  printf("fd2:%d\n",fd2);  printf("fd3:%d\n",fd3);  printf("fd4:%d\n",fd4);  printf("fd5:%d\n",fd5);  close(fd1);close(fd2);close(fd3);close(fd4);close(fd5);return 0;
}

测试结果如下:

         这些数字其实是一个数字的下标,在PCB中有一个指针数组 (称为文件描述符表)。这个指针数组存放的是指向struct_file这个文件管理的内核数据结构。

        进程通过fd这个数组下标就能够访问文件结构体!

具体关系可见下图:

 

 五. 下篇重点: 文件fd, Linux下一切皆文件

http://www.lryc.cn/news/497553.html

相关文章:

  • 【Python网络爬虫笔记】8- (BeautifulSoup)抓取电影天堂2024年最新电影,并保存所有电影名称和链接
  • Rancher V2.7.0安装教程
  • STM32MX 配置CANFD收发通讯
  • (12)时间序列预测之MICN(CNN)
  • 嵌入式蓝桥杯学习3 外部中断实现按键
  • 自由学习记录(29)
  • 使用YOLO系列txt目标检测标签的滑窗切割:批量处理图像和标签的实用工具
  • 架构10-可观测性
  • git管理Unity项目的正确方式
  • openssl使用哈希算法生成随机密钥
  • 将word里自带公式编辑器编辑的公式转换成用mathtype编辑的格式
  • 校园失物招领系统基于 SpringBoot:点亮校园归还遗失物之光
  • dhcpd服务器的配置与管理(超详细!!!)
  • Qml之基本控件
  • 【Java从入门到放弃 之 Stream API】
  • Ruby On Rails 笔记1——Rails 入门
  • 高效开发 Python Web 应用:FastAPI 数据验证与响应体设计
  • 基于“开源 2+1 链动 O2O 商城小程序”的门店拉新策略与流程设计
  • 33.5 remote实战项目之设计prometheus数据源的结构
  • 微服务springboot详细解析(一)
  • 深入探讨Go语言中的双向链表
  • Fastapi + vue3 自动化测试平台---移动端App自动化篇
  • ElasticSearch easy-es 聚合函数 group by 混合写法求Top N 词云 分词
  • 在 ASP.NET C# Web API 中实现 Serilog 以增强请求和响应的日志记录
  • 2024年顶级小型语言模型前15名
  • 精通 Python 网络安全(一)
  • 【python自动化二】pytest集成allure生成测试报告
  • 网络版本的通讯录青春版(protobuf)
  • 开源模型应用落地-安全合规篇-用户输入价值观判断(三)
  • 神经网络入门实战:(十四)pytorch 官网内置的 CIFAR10 数据集,及其网络模型