目录
🌈前言
本篇文章进行操作系统中文件描述符的学习!!!
🌷1、文件的概念
概念:
-
文件 = 文件内容 + 文件属性,文件属性也是数据,即使我们创建一个空文件,也要占据磁盘空间
-
文件操作 = 文件内容的操作 + 文件属性的操作 – 在操作文件的过程中,有可能既改变了文件的内容,又改变了文件属性。比如:改变内容到一定次数,文件的时间属性也会被修正
-
所谓的“打开”文件,就是将文件的内容和属性加载到内存当中!!! – 冯诺依曼体系结构
-
是不是所有的文件,都会处于被打开状态?绝对不是!没有被打开的文件,在磁盘上静静的存储着
打开的文件(“内存文件”)和“磁盘文件”的区别是什么???
- 内存文件:磁盘文件中的内容和属性大部分被加载到进程当中,就叫做“内存文件”
- 磁盘文件:存储在磁盘中的文件
- 区别:一个是虚拟内存文件,一个是磁盘文件
- 通常我们使用C语言打开文件、访问文件和关闭文件,都是通过fopen、fclose、fread、fwrite…函数来进行操作的!!!
是谁在进行文件相关操作的呢??? 答案是:进程
-
一个程序编译链接好后,会生成一个可执行程序,这时程序还没有被执行,在磁盘中还是一个普通的文件!!!
-
该程序运行时,会被加载到内存当中,然后程序的代码和数据被进程读取,进程会被加入到运行队列
-
当被调度器调度时,会执行进程中程序的代码,这时就会执行相应的文件操作和其他代码
🌹2、文件操作(C语言)
🍡2.1、概念+基本打开关闭操作
文件在读写之前应该先打开文件,在使用结束之后应该关闭文件 <stdio.h>
//打开文件
FILE * fopen ( const char * filename, const char * mode );
//关闭文件
int fclose ( FILE * stream );
-
函数 fopen 打开文件名为 “filename” 指向的字符串的文件,将一个流与它关联
-
当文件被打开时,会默认打开三个流,分别是:stdin & stdout & stderr(后面理解)
-
仔细观察发现,这三个流的类型都是FILE*, fopen返回值类型,文件指针
-
如果filename不存在,则会在”当前路径下“创建新的文件!!!
如何理解”当前路径“???
- 我们都知道程序执行后,会被加载到内存,然后被进程读取
- 当CPU执行到打开文件时,发现文件不存在
- 那么CPU就会在”进程所处的工作路径下创建新的文件“!!!
查看进程中的cwd属性,ls /proc/进程pid
[lyh_sky@localhost test]$ cat myfile.c
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
int main()
{
// 以写的方式打开文件,不存在就新建一个文件
FILE* fp = fopen("log.txt", "w");
if (fp == NULL)
{
perror("fp");
}
// 让该进程一直在运行,方便查看cwd
printf("我是一个进程,我的pid: %d\n", getpid());
while (1)
{
}
// 关闭文件
fclose(fp);
return 0;
}
[lyh_sky@localhost test]$ ls
log.txt makefile myfile myfile.c // 在当前工作目录创建了新的文件
用chdir系统函数修改进程当前所处工作路径,并且在新路径创建文件
[lyh_sky@localhost test]$ cat myfile.c
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
int main()
{
chdir("/home/lyh_sky");
// 以写的方式打开文件,不存在就新建一个文件
FILE* fp = fopen("log.txt", "w");
if (fp == NULL)
{
perror("fp");
}
// 让该进程一直在运行,方便查看cwd
printf("我是一个进程,我的pid: %d\n", getpid());
while (1)
{
}
// 关闭文件
fclose(fp);
return 0;
}
// 运行后,然后ctrl + c退出进程,就能看到文本文件存在了
[lyh_sky@localhost test]$ ls /home/lyh_sky -al | grep log.txt
-rw-rw-r--. 1 lyh_sky lyh_sky 0 11月 20 23:58 log.txt
🍢2.2、文件的打开方式
- 参数 “mode” 指向一个字符串,以下列序列之一开始 (序列之后可以有附加的字符):
打开方式 | 作用 | 指定文件不存在 |
---|---|---|
“r” | 打开文本文件,用于读。流被定位于文件的开始 | 出错 |
“r+” | 打开文本文件,用于读写。流被定位于文件的开始 | 出错 |
“w” | 将文件长度截断为零,或者创建文本文件,用于写。流被定位于文件的开始 | 建立新的文件 |
“w+” | 打开文件,用于读写。如果文件不存在就创建它,否则将截断它。流被定位于文件的开始 | 建立新的文件 |
“a” | 打开文件,用于追加 (在文件尾写)。如果文件不存在就创建它。流被定位于文件的末尾 | 建立新的文件 |
“a+” | 打开文件,用于追加 (在文件尾写)。如果文件不存在就创建它。读文件的初始位置是文件的开始,但是输出总是被追加到文件的末尾 | 建立新的文件 |
实例:以写的方式打开文件,并且关闭文件
#include <stdio.h>
int main()
{
// 以写的方式打开文件,不存在就新建一个文件
FILE* fp = fopen("log.txt", "w");
// 判断是否打开成功
if (fp == NULL)
{
perror("fp");
}
// 关闭文件
fclose(fp);
return 0;
}
注意:当以w的方式打开文件时,如果文件有数据,会被截断清空,并且流被定位到文件的开始!
🍣2.3、文件的读写操作
文件的顺序读写
功能 | 函数名 | 适用于 |
---|---|---|
字符输入函数 | fgetc | 所有输入流 |
字符输出函数 | fputc | 所有输出流 |
文本行输入函数 | fgets | 所有输入流 |
文本行输出函数 | fputs | 所有输出流 |
格式化输入函数 | fscanf | 所有输入流 |
格式化输出函数 | fprintf | 所有输出流 |
二进制输入 | fread | 文件 |
二进制输出 | fwriter | 文件 |
例子一:对文件进行写入数据 – 使用”fprintf“
int fprintf(FILE *stream, const char *format, …);
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int main()
{
// 以写的方式打开文件,不存在就新建一个文件
FILE* fp = fopen("log.txt", "w");
if (fp == NULL)
{
perror("fp");
}
int cnt = 1;
while (cnt <= 5)
{
fprintf(fp, "%d: hello world!!!\n", cnt);
++cnt;
}
// 关闭文件
fclose(fp);
return 0;
}
例子二:对文件进行读取数据 – 使用”fgets“
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
int main()
{
// 以读的方式打开文件,不存在会报错
FILE* fp = fopen("log.txt", "r");
if (fp == NULL)
{
perror("fp");
}
// 读取文件数据
char buffer[64];
while(fgets(buffer, sizeof(buffer), fp) != NULL)
{
printf("%s", buffer);
}
// 关闭文件
fclose(fp);
return 0;
}
例子三:对文件进行追加数据 – 使用“fprintf”进行写入
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
int main(