Linux文件系统调用接口

文件=内容+属性

所有对文件的操作就是对 1.文件内容  2.文件属性。

内容是数据,属性也是数据,存储文件,必须既存储内容,也要存储属性。

文件没有被访问时,一般在磁盘中。对文件访问时,由冯诺依曼体系结构知,需要将文件加载到内存中,才能被操作。

加载磁盘上的文件到内存中,是由OS来完成,必然存在许多的文件等待被加载,操作系统需要对这些已经加载文件的管理。管理的方式就是先描述、在组织

描述组织方法

描述:创建文件描述结构体 struct file{属性   文件指针  }

组织:将文件描述的结构体,以某种数据结构链接。对文件的管理,就转化为对结构体的增删查改。

C语言接口

回忆完文件的预备知识后,就来回忆一下C语言的文件操作IO接口

文件操作函数    功能
fopen    打开文件
fclose    关闭文件
fputc    写入一个字符
fgetc    读取一个字符

fputs    写入一个字符串
fgets    读取一个字符串
fprintf    格式化写入数据
fscanf    格式化读取数据
fwrite    向二进制文件写入数据
fread    从二进制文件读取数据
fseek    设置文件指针的位置
ftell    计算当前文件指针相对于起始位置的偏移量
rewind    设置文件指针到文件的起始位置
ferror    判断文件操作过程中是否发生错误
feof    判断文件指针是否读取到文件末尾

fopen()

FILE* fopen(const char* path ,const char* mode)

函数参数为  文件路径  和打开方式

返回参数为文件指针,打开成功返回指针,失败返回NULL

fputs()

int fputs(const char *s ,FILE*stream)

将字符串写入特定流中,成功则返回字符串个数,失败返回-1

fprintf()

int fprintf(FILE*stream ,const char* format ,......)

向某个流中,格式化写入

fopen()的写入方式

文件使用方式    含义    如果指定文件不存在
"r"(只读)    为了输入数据,打开一个已经存在的文本文件    出错
"w"(只写)    为了输出数据,打开一个文本文件            建立一个新的文件
"a"(追加)    向文本文件尾添加数据                               建立一个新的文件
"rb"(只读)    为了输入数据,打开一个二进制文件        出错
"wb"(只写)    为了输出数据,打开一个二进制文件       建立一个新的文件
"ab"(追加)    向一个二进制文件尾添加数据                    出错
"r+"(读写)    为了读和写,打开一个文本文件                 出错
"w+"(读写)    为了读和写,建立一个新的文件               建立一个新的文件
"a+"(读写)    打开一个文件,在文件尾进行读写            建立一个新的文件
"rb+"(读写)    为了读和写打开一个二进制文件              出错
"wb+"(读写)    为了读和写,新建一个新的二进制文件    建立一个新的文件
"ab+"(读写)    打开一个二进制文件,在文件尾进行读和写    建立一个新的文件

部分演示:
1、写"w"形式打开一个文件

效果等同 >

  1 #include<stdio.h>2 #include<unistd.h>3 4 int main()5 {6   FILE* fp=fopen("log.txt","w");7   if(fp==NULL)8   {9     perror("open file");10     return 1;11   }12   const char* msg="hello world";13   int cnt=1;14   while(cnt<=10)15   {16     fprintf(fp,"%s:%d\n",msg,cnt);                                                                          17     cnt++;18   }19   fclose(fp);20   return 0;21 }
~

上述代码以w(只写权限)打开一个文件,在当前路径低下,没有存在log.txt文件,w形式会创建一个名为log.txt的空文件。对文件的操作是写入10条hellow world

编译运行后

会在当前路径底下生成log.txt文件  利用cat重定向,将文件内容打印到显示器上


2、在"a"追加形式打开

效果等同于  >>

将1.的代码做修改,在将”w“修改成a ,在循环体中,打印内容为hellow linux

在第一步的log.txt文件下已存在内容的情况下,在编译运行。

"a"是追加,则不会清空文件,cat重定向到显示器后

三个默认流

Linux下的经典话是“一切皆文件”。

在linux下显示器和键盘也能看作文件。我们在显示器上看到文件内容,实际上就是往"显示器文件"写入文件。电脑获得我们在键盘敲击的字符,实际上就是往"键盘文件"读取了数据。

从刚才我们的回顾,就可以发现既然是从显示器文件中写入数据,那么文件一定是被打开的。


结论


任何进程运行时,都会打开三个输入输出流,即标准输入,标准输出,和标准错误流。

标准输入的默认是键盘。标准输出和错误默认是显示器。

在C语言中的三个默认流,是stdin ,stdout ,stderr 

在C++上也存在三个流,对应cin ,cout ,cerr

系统接口

在我们用的C语言文件接口,一定是封装了系统调用接口!

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);

open函数

pathname为要打开或者创建的文件名

如果是pathname给出路径,则在该路径下创建

如果是文件名,则在当前文件所在路径下创建名为pathname的文件

flags: 打开文件时,可以传入多个参数选项,用下面的一个或者多个常量进行“”运算,构成

位图的操作

参数含义
O_RDNOLY只读方式打开 1
O_WRNOLY只写方式打开 2
O_RDWR读写方式打开 3  三者必须指定一种
O_CREAT

文件不存在,就创建文件

O_APPEND追加写
返回值:成功返回新的文件描述符,失败返回-1

对flags的说明  flags是位图的形式  每一个选项的二进制序列只有一个1   在运算中,我们可以通过或运算  open函数内部通过与运算得到每一个选项

举例:

想以读的方式打开文件,在没有文件存在就创建文件  通过 | 运算符

open(pathname,O_RDNOLY | O_CREAT)

下面通过代码验证:创建一个log.txt(不存在) 以只写的方式打开

  1 #include<stdio.h>                                                                                           2 #include<sys/stat.h>3 #include<fcntl.h>4 5 6 int main()7 {8   int fd = open("log.txt",O_WRONLY|O_CREAT);9   if(fd<0)10   {11     perror("open fail");12     return -1;13   }14   printf("fd 是%d\n",fd);15   return 0;16 }

编译运行后,在当前路径下得到log.txt

得到fd 是 3  我们关心fd为什么会是3,而不是0 1  2呢?在下面将会谈到!

谈一下mode参数

mode参数是文件默认权限的16进制

0666代表 u g o 对应所有者 所属组 其它  rwx  011就是6对应 wx权限 

创建文件的时候得到是0664 受到权限掩码的影响,权限掩码是0002 


2、close () 

用于关闭文件 ——fclose就是封装了close

通过查阅man手册,了解close的用法

close()的参数是 fd(open被打开文件的返回值)

成功关闭文件返回0,失败返回-1


3、write()

查阅2号手册

       #include <unistd.h>ssize_t write(int fd, const void *buf, size_t count);

fd 是文件描述符

buff表示缓冲区

count字节数

write是将缓冲区count大小的数据写入fd中。

返回值是一个可为负的值,表示实际写入多少字节。

4、read()

       #include <unistd.h>ssize_t read(int fd, void *buf, size_t count);

read与write类似

fd是文件描述符

buff是缓冲区

count是字节数

read是从缓冲区读取count字节大小值到fd中

注意:

读取缓冲区的大小是strlen(s )  需要+1吗?

不需要。如果strlen(s)+1  则再打开log.txt文件时会出现乱码,因为\0是C语言的规定,在文件中没有要求。会被OS译为乱码。

演示

利用snprintf函数将hollow Linux写入缓冲区buff,write函数从buff中读取数据,并写入到log.txt文件

cat重定向到显示器

  1 #include<stdio.h>2 #include<sys/stat.h>3 #include<fcntl.h>4 #include<string.h>5 #include<unistd.h>6 int main()7 {8   umask(0);9   int fd = open("log.txt",O_WRONLY|O_CREAT,0666);10   if(fd<0)11   {12     perror("open fail");13     return -1;14   }15   printf("fd :%d\n",fd);16   const char* s="hellow Linux!";17   int cnt=1;18   while(cnt<=5)19   {20     char buff[128];21     snprintf(buff,sizeof(buff),"msg:%s,cnt:%d\n",s,cnt);                                                    22     write(fd,buff,strlen(buff));23     cnt++;24   }25   close(fd);26   return 0;27 }
~

read的函数用法与write类似,就不做过多介绍。


注意:O_WRONLY|O_CREAY 默认是不会对文件内容清空的,如果需要清空,就要加上O_TRUNC

这时候我们的open调用就非常类似C语言的fopen()


这时,我们就可以猜测C语言文件就是封装了系统调用接口。

OS不信任用户,OS就给用户提供系统调用,程序员通过C语言调用文件接口,必然间接调用系统调用。

读文件本质是将把文件从磁盘加载到内存中,这是冯诺依曼体系规定的。

写文件意味着可能修改文件,必然也要先把文件加载到内存中。

用户不能直接管理硬件,OS是硬件的直接管理者。


回忆文件,内容加属性。

要访问文件必须先把文件加载到内存。OS对文件的管理就是先描述再组织。

C语言必定封装了某种系统调用。

open的pathname是文件名,flag是一个32位的位图,在传参时通过或运算添加。

write和read非常的类似。从缓冲区buff写入/读到 fd对应的文件中。

O_WRNOLY|O_CREAY|O_TRUNC是清空文件的写入。


下面我们还有问题?

FILE是封装了fd吗

fd为什么默认从3开始

Linux下一切皆文件是为什么

重定向是如何理解


在下文将会详细介绍

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/313549.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

第三十七周周报:文献阅读+掩码、多头注意力机制+位置编码

目录 摘要 Abstract 文献阅读&#xff1a;基于注意力的LSTM大地震预报网络 现有问题 提出方法 基于注意力的LSTM网络 研究实验 实验目的 数据集 评估指标 数据预处理和特征提取 结果讨论 Masked Self-Attention&#xff08;掩码自注意力&#xff09; Muti-Head S…

Git:常用命令(一)

取得项目的Git 仓库 从当前目录初始化 1 git init 初始化后&#xff0c;在当前目录下会出现一个名为.git 的目录&#xff0c;所有Git 需要的数据和资源都存放在这个目录中。不过目前&#xff0c;仅仅是按照既有的结构框架初始化好了里边所有的文件和目录&#xff0c;但我们还…

第二节 linux操作系统安装与配置

一&#xff1a;Vmware虚拟机安装与使用   ①VMware是一个虚拟PC的软件&#xff0c;可以在现有的操作系统上虚拟出一个新的硬件环境&#xff0c;相当于模拟出一台新的PC &#xff0c;以此来实现在一台机器上真正同时运行多个独立的操作系统。   ②VMware主要特点&#xff1a…

【代码随想录】刷题笔记Day42

前言 这两天机器狗终于搞定了&#xff0c;一个控制ROS大佬&#xff0c;一个计院编程大佬&#xff0c;竟然真把创新点这个弄出来了&#xff0c;牛牛牛牛&#xff08;菜鸡我只能负责在旁边喊加油&#xff09;。下午翘了自辩课来刷题&#xff0c;这次应该是元旦前最后一刷了&…

流逝的时光

文章目录 创作历程展望2024 创作历程 自2019.6.28注册csdn&#xff0c;期间断断续续的通过其查找相应资料&#xff0c;受益颇多 今研一&#xff0c;发现论文看了又忘&#xff0c;于是借此平台来记录&#xff0c;可以看到基本都是基于原论文进行翻译&#xff0c;并没有所思所想&…

go slice源码探索(切片、copy、扩容)和go编译源码分析

文章目录 概要一、数据结构二、初始化2.1、字面量2.2、下标截取2.2.1、截取原理 2.3、make关键字2.3.1、编译时 三、复制3.1、copy源码 四、扩容4.1、append源码 五&#xff1a;切片的GC六&#xff1a;切片使用注意事项七&#xff1a;参考 概要 Go语言的切片&#xff08;slice…

付费进群系统源码带定位完整独立版(12月30日)再次修复首发

搭建教程 nginx1.2 php5.6–7.2均可 最好是7.2 第一步上传文件程序到网站根目录解压 第二步导入数据库&#xff08;shujuku.sql&#xff09; 第三步修改/config/database.php里面的数据库地址 第四步修改/config/extra/ip.php里面的域名 第四步设置伪静态thinkphp 总后台账号&…

FPGA项目(13)——基于FPGA的电梯控制系统

1.摘要 随着科技的发展&#xff0c;电梯早在上个世纪就已进入人们的生活。对于电梯的控制&#xff0c;传统的方法是使用继电器——接触器控制系统进行控制。随着EDA技术的发展&#xff0c;FPGA已广泛应用于各项电子设计中&#xff0c;本设计即利用FPGA来实现对电梯控制系统的设…

传感器原理与应用复习--超声波、微波、红外及热电偶传感器

文章目录 上一篇超声波传感器微波传感器红外传感器热电偶传感器下一篇 上一篇 传感器原理与应用复习–光电式与半导体式传感器 超声波传感器 超过2万赫兹以上的波称为超声波 压电式超声波探头常用材料是压电晶体和压电陶瓷。它是利用压电材料的压电效应来工作的。 逆压电效…

redhat 8 安装openstack

redhat 8 安装openstack 1、安装文档2、redhat 8 安装openstack3、使用openstack 1、安装文档 openstack官方安装文档 https://docs.openstack.org/install-guide/ 2、redhat 8 安装openstack 3、使用openstack

王道考研计算机组成原理——存储系统

存储系统的基础知识 微信打开的时候会有一个人站在地球上&#xff0c;这个过程就是把程序从辅存转移到主存&#xff0c;数据只有调入主存当中才可以被CPU访问 cache&#xff1a;主存速度还是慢&#xff0c;为了进一步缓解CPU和主存之间的速度矛盾 在微信打视频聊天的时候&am…

Apalis_iMX6_eMMC常驻ToradexEasyInstaller

By Toradex胡珊逢 简介 Toradex Easy Installer 是 Toradex 计算机模块上出厂预装的软件&#xff0c;可以令开发人员首次安装不同的操作系统&#xff0c;同时也适生产线上模块的批量烧写作业。通常在安装系统后&#xff0c;Toradex Easy Installer 将被从模块的存储上擦除。本…