文章目录
前言:本文探讨的是“打开的”文件。
预备知识
如进程一样,文件 = 文件属性 + 文件内容
文件属性包括:文件的大小,文件的创建日期,文件的拥有者和所属组等等。
文件内容就是你对文件写入的内容。
对文件的操作 = 对文件属性的操作 + 对文件内容的操作
访问文件的本质是进程在访问文件
而进程访问文件是需要接口的,进程访问文件是通过系统调用接口,而我们平时使用的访问文件的接口是语言接口。
语言接口就是对系统调用接口的封装。
语言接口有很多,(C语言有C语言的接口,java有java的接口……)但是他们最终都能访问到磁盘上的文件,这是为什么呢?
难道磁盘为了这些语言而专门写了这些接口吗?显然不是,是系统调用的接口是只有一个 ,后续语言就是对系统调用进行封装。
所以,看似各种语言使用的不同的接口,其实用的同一套底层接口。
C语言文件接口
fopen()函数
查看C语言的打开文件接口
对于fopen()函数
作用:打开文件/创建文件
参数
- const char* path: 打开文件的路径/创建文件的路径
- const char* mode: 对文件进行的操作
对于mode,可以看文档
mode | 说明 |
---|---|
r | 读模式,允许读取文件 |
r+ | 读写模式,可以同时读,也可以同时写。(原有文件内容都会被保存。) |
w | 清空写模式,如果文件不存在,则自动创建。如果文件已经存在,把原有的文件内容删除光。再写入新的内容。 |
w+ | 清空读写模式,读新写入的内容。 |
a | 就是append,在文件最后追加新写入的内容。文件不存在就创建新文件 |
有这些操作的内容,接下来通过示例演示以下:
观察结果:
本程序是在/home/zyy/Linux-learning/24-02-04-lesson/中执行的,所以在这里创建了一个进程。
我们创建文件的路径是/home/zyy/Linux-learning/,所以在该路径下也创建了一个进程。
可以观察到两个进程的pid是不一样的,说明我们在运行自己创建的程序的时候,创建了一个进程,在程序运行本身调用了创建/打开文件函数,也创建了一个进程。
w方式
#include<stdio.h>
2 #include<string.h>
3
4 int main()
5 {
6 FILE* fp = fopen("test.txt", "w");
7 if(fp == NULL)
8 {
9 return 1;
10 }
11
12 const char* s = "hello in message";
13 fwrite(s, strlen(s), 1, fp);
14 fclose(fp);
15 return 0;
16 }
a方法
1 #include<stdio.h>
2 #include<string.h>
3
4 int main()
5 {
6 FILE* fp = fopen("test.txt", "a");
7 if(fp == NULL)
8 {
9 return 1;
10 }
11
12 const char* s = "hello in message\n";
13 fwrite(s, strlen(s), 1, fp);
14 fclose(fp);
15 return 0;
16 }
fwrite()
功能: 向指定的文件中写入若干数据块,
如成功执行则返回实际写入的数据块数目。
const void *ptr 指向要被写入的数组元素的指针
size_t sizoe 表示每个元素的大小
size_t nmemb 表示要写入多少个这种元素
FILE* stream 表示fopen()打开文件之后返回的文件指针
使用方法:
fprintf()
功能: 格式化输出到一个流文件中(把s输出到fp中)
FILE* stream 文件指针,fopen()打开的那个文件返回的文件指针
const char *format 格式化输出
…… 可变参数列表
使用方式:
执行结果:
fputs()
功能: 向指定的文件写入一个字符串(不自动写入字符串结束标记符‘\0’)
const char *s 写入的字符串
FILE *stream 文件指针
fread()
功能: 从文件中读取若干字节数据到内存缓冲区中
const void *ptr 将文件中的二进制数据读取到该缓冲区中
size_t sizoe 读取的基本单元,单位是字节
size_t nmemb 读取的基本单元的个数
FILE* stream 表示fopen()打开文件之后返回的文件指针
使用方式:
结果:
fgets()
功能:从 stream 流中读取 size 个字符存储到字符指针变量 s 所指向的内存空间。
char *s: 要存到的内存空间的首地址
int size: 读取字符串的长度
FILE *stream: 从何种流中读取,可以是标准输入流stdin
使用:
结果:
标准输入输出流
执行程序时,C语言会默认打开三个标准输入输出文件。
- stdin 标准输入
- stdout 标准输出
- stderr 标准错误
系统调用接口
功能:打开/创建一个文件或者设备
const char *pathname: 要打开的文件的路径
int flages: 标志位(以读/写的方式打开)
code_t mode: 权限设置
标志位
第二个参数是标志位,告诉open函数是要以读还是以写的方式打开文件,第二个参数必须包含O_RDONLY, O_WRONLY, 以及 O_RDWR 中的一个。
- O_WRONLY:以写的方式打开文件
- O_RDONLY:以读的方式打开文件
- O_WRONLY:以读写的方式打开文件
在此基础上可以附加其他参数
- O_CREAT:当文件不存在时,创建文件
- O_APPEND:在文件末尾追加
问题1:只有一个参数,如何同时传多个参数呢?
问题2:为什么参数类型是int?两个问题可以同时回答。
32位下有 32个bit位,我们以低8位为例。
假设第一位表示以写的方式打开: 0000 0001 O_WRONLY
假设第二位表示以读的方式打开: 0000 0010 O_RDONLY
假设第三位表示文件不存在时创建: 0000 0100 O_WRONLY
而如果我想文件不存在时创建文件并且以写的方式打开,就可以使用按位或的方式 :
0000 0001
0000 0100
————————
0000 0101
结果: int flag = O_WRONLY|O_CREAT
文件权限
文件权限分为三部分:用户权限,组权限,其他权限
示例是644,为8进制表示。
用2进制表示:示例中-rw -rw r-- 表示为 011 011 100 ,转化为8进制就是 664。
我们一般设置权限为666。
返回值是int类型的文件描述符
使用:
结果:
fclose()
传入文件描述符即可。
作用:关闭文件。
write()
功能: 向fd中写入字节数为count的buf内容
int fd 文件描述符,标识要写入的文件
void *buf 要写入的内容
size_t count 向文件中写入多少字节
结果:
如果更换写入内容:
结果:
不会对内容进行清空,会对内容进行覆盖
如果想清空需要加上O_TRUNC
结果:
read()
功能:将fd中的内容读到buf中,读count个字节
用法:
向test.txt文件中写入的内容:
结果: