1.C语言中的文件操作函数
文件的打开
data:image/s3,"s3://crabby-images/f3e35/f3e356d8387526d2313d5534de992282d4b3ea6b" alt=""
path为文件路径,mode为打开方式,它们都是字符串。
代码演示:
data:image/s3,"s3://crabby-images/75b7a/75b7a5f94d9e9d2cad8810a4e09697ab143f4286" alt=""
此时,当前目录中并没有log.txt文件,但是没关系,fopen会在当前路径下创建log.txt文件。
data:image/s3,"s3://crabby-images/c243c/c243c89686ca15092c6539cddbde9a57d766244a" alt=""
简单来说,当前路径指的是一个进程运行起来的时候,每个进程都会去记录自己当前所处的工作路径。所以当前路径也就是当前进程的工作路径。
下面来验证一下:
data:image/s3,"s3://crabby-images/83e78/83e780a83e21678474236ed51ca1384af15ecb90" alt=""
data:image/s3,"s3://crabby-images/4a74f/4a74f7ed640262dd11a2c57ed68e5cb1488aeea4" alt=""
注∶使用fopen时,填写mode参数,单纯以w方式打开文件,会自动清空文件原有的数据。r+(读写)代表文件不存在则出错, w+(读写)代表文件不存在则创建。(带有+的表示读写)。a代表向文件中追加内容。
打开文件方式的总结:
data:image/s3,"s3://crabby-images/07997/079970da4fe740dd12c0e570aa6faebbc4333bd2" alt=""
文件的关闭
data:image/s3,"s3://crabby-images/39bcb/39bcbd37d3238d10fbc315d7543c2bc671414da5" alt=""
C语言中的其它读写文件函数
int fputs(const char *s, FILE *stream);
char *fgets(char *s, int size, FILE *stream);int fprintf(FILE *stream, const char *format, ...);
int fscanf(FILE *stream, const char *format, ...);
2.系统文件IO操作函数
1.open
data:image/s3,"s3://crabby-images/1a33e/1a33e28c4b726ed245b9dd0e06e7cb08c76142d2" alt=""
pathname:打开文件名
flags :标志位。(打开文件时,可以传入多个参数选项,用一个或者多个常量进行"或"运算,构成 flags)O_RDONLY:只读打开
O_WRONLY:只写打开
O_RDWR:读写打开
o_CREAT:若文件不存在,则创建它。需要使用mode选项,来指明新文件的访问权限O_APPEND:追加写
o_TRUNC:打开时,清空文件内容。
返回值: 成功则返回新打开的文件的文件描述符失败:返回-1 ,并且errno被设置。
RETURN VALUE
open() and creat() return the new file descriptor, or -1 if an error occurred (in which case, errno is set appropriately).
flag如何传递多个选项,是通过比特位来传递选项的。
代码演示:
data:image/s3,"s3://crabby-images/8d879/8d87923093e48529e07353ff1d56e6605863c9f8" alt=""
data:image/s3,"s3://crabby-images/88ac4/88ac4e078cede6de7f2ab1dddbe5c294d740e003" alt=""
因此,我们可以使用|(或)来帮助我们传递多个参数,以此实现不同的功能。
mode参数
如果你使用O_CREAT参数创建一个新的文件,那么你还可以通过第三个参数mode来设置该文件的权限。
2.close
data:image/s3,"s3://crabby-images/aa96f/aa96fac5a0ab848c5f76602ac4236b286193844e" alt=""
3.read
data:image/s3,"s3://crabby-images/4a255/4a25583fe84c1f3c5aa60342c9b2e6b1925ce741" alt=""
fd:要读取的文件
buf:存放读取内容的数组
count:读取的内容大小
4.write
data:image/s3,"s3://crabby-images/fcc70/fcc70b58fee75825d415bcd1a9ddb3be1e7ca820" alt=""
fd:要写入的文件
buf:要写入的内容
count:所写内容的大小。
3.文件描述符fd
文件描述符的引入
代码演示:
data:image/s3,"s3://crabby-images/b665c/b665c4991a83314ff7e6e9096b041d9dd09ec75d" alt=""
运行结果:
data:image/s3,"s3://crabby-images/68324/68324f8bb558ba37bdb33a2efe3d53a006436f28" alt=""
fd是一个整数,我们发现,结果是从3开始的,为什么是从3开始的呢?
在C语言阶段,我们知道在程序运行时,操作系统会默认打开三个标准输入输出流:标准输入,标准输出,标准错误。对应到C语言当中就是stdin、stdout以及stderr。在C++中则是cin、cout、cerr。
我们知道C语言中的stdin、 stdoutl以及stder这三个家伙实际上都是FILE*类型的,并不是int类型。是因为FILE*是一个结构体指针,对fd进行了封装,而在Linux层面,只认fd。
代码验证:
data:image/s3,"s3://crabby-images/fbb7e/fbb7e827e1820e11db9491de8046d52c73f890dd" alt=""
结果如下:
data:image/s3,"s3://crabby-images/d730a/d730acb2355e5f0e9c29f37ef96612acd3832f2c" alt=""
对fd的理解
进程要访问文件,必须先打开文件,一个进程可以同时打开多个文件,操作系统中有大量的进程,也就是说操作系统要同时打开大量的文件,操作系统要对这些打开的文件进行管理,方式:先描述,再组织。
上边,我们发现fd是从0,1,2,3,4,.........
我们在哪里将从0开始的连续增长的整数--------数组下标。
其实,文件描述符的本质就是数组下标。
data:image/s3,"s3://crabby-images/be5a2/be5a29560568132b737ae646571f63589d0011d7" alt=""
对应的内核部分代码:
data:image/s3,"s3://crabby-images/07a55/07a557de03cb56651a0146ee6231f0e40c657e40" alt=""
4.文件描述符的分配规则
代码演示:
data:image/s3,"s3://crabby-images/af60a/af60ae26161e1aea0e085c50b982a02b1cd0ad8d" alt=""
data:image/s3,"s3://crabby-images/f0745/f074500fecb6c1a7f5b8d461e866b6d47f465f79" alt=""
如果先关闭了0号文件描述符呢?
data:image/s3,"s3://crabby-images/4ed50/4ed50322c2e511299ec8893793ce6daf64ab72c7" alt=""
data:image/s3,"s3://crabby-images/52dfd/52dfdff21c5fe78d9ac24c85c35ecb8da0d7e8e6" alt=""
结论:
文件描述符是从最小并且没有被使用的开始分配的。
5.重定向
重定向的原理
(以输出重定向为例)
data:image/s3,"s3://crabby-images/420fb/420fb2f121e3f16afd3734c3e0e4bbf73642ad1e" alt=""
输出重定向的本质:
更改进程指向的files_struct内的struct file* fd_array[]数组内下标1内指向的struct file对象的地址。
重定向对应的系统调用
dup2
data:image/s3,"s3://crabby-images/34ba2/34ba211f1fc98e2fadec4ee12e26789a723fa607" alt=""
RETURN VALUE(返回值)
On success, these system calls return the new descriptor. On error, -1 is returned, and errno is set appropriately.
成功则返回新的文教描述符; 失败则返回-1,并且errno被设置。
代码演示:
data:image/s3,"s3://crabby-images/39216/3921613652701313924cc2cc2a23f2894179f6bb" alt=""
结果:
data:image/s3,"s3://crabby-images/f50f7/f50f77212a27b271f4438567aeed45276fd81a91" alt=""
关于dup2(int oldfd,int newfd)两个参数的填写
以上边的代码为例,目的是将本该打印到显示器的内容显示到log.txt文件中,结合对文件描述符的理解(文件描述符就是数组下标,struct file* fd_array[]数组内填写的是struct file对象的地址),再结合2号手册
data:image/s3,"s3://crabby-images/52e19/52e19d91ae4ba5ea75a5645022ede0c366db0e08" alt=""
我们不难理解到,我们是要把地址为fd对应的struct file地址拷贝到地址为1对应的位置,所以,要重定向的目标文件描述符在后(newfd), 要进行重定向的文件描述符在前(oldfd)。
6.理解Linux下一切皆文件
data:image/s3,"s3://crabby-images/a43b4/a43b43252a6ad31d0fbbd7e2680bc9774b5d8183" alt=""
底层不同的硬件,一定对应不同的操作方法,但是OS管理这些底层硬件,使用统一的结构来进行管理的,在OS视角,底层硬件没有任何区别。