库IO函数的工作流程
使用fopen函数打开一个文件,返回的是一个FILE *fp指针,指针指向三个重要组成成员:
- 文件描述符:描述符中定义了文件的Inode,通过Inode可以找到对应的数据块
- 文件指针:读和写共享一个文件指针,读或者写都会引起文件指针的变化
- 文件缓冲区:减少对磁盘的读写次数。
在linux中 /usr/include 文件夹中找到 stdio.h的第48行
找到 typedef struct IO_FILE FILE; //FILE *p
在libio.h的第241行,定义了 struct_IO_FILE, 这个结构体定义中只有一个fileno成员,就是文件描述符
C库函数与系统函数的关系
write函数的定义,write函数不是系统调用,是提供给用户的一个接口。
printf函数也返回一个FILE*
虚拟地址空间
上图也可以进一步理解内存四区:代码区,全局区,堆区,栈区
上图是进程的虚拟地址空间,内核区中的很重要的是进程管理,进程管理中有一个区域是PCB(本质是一个结构体),PCB中有文件描述符,文件描述符中存放着打开的文件描述符,涉及到文件的IO操作都会用到这个文件描述符。
PCB和文件描述符表
虚拟地址空间——>内核区——>PCB——>文件描述表——>文件描述符——>文件IO操作使用文件描述符
C语言与linux中区别:其实都是操作文件描述符
FILE *fp = fopen();
int fd =open();
程序实例:
熟悉简单的 open,write,lseek read close操作
1 #include<stdio.h>
2 #include<stdlib.h>
3 #include<string.h>
4 #include<sys/types.h>
5 #include<unistd.h>
6 #include<sys/stat.h>
7 #include<fcntl.h>
8
9 int main(int argc, char *argv[])
10 {
11 //打开文件
12 int fd = open(argv[1],O_RDWR | O_CREAT, 0777);
13 if(fd<0)
14 {
15 perror("open error");
16
17 return -1;
18 }
19
20 //写入文件
21 write(fd,"hello world",strlen("hello world"));
22
23 //移动文件指针到文件开始处
24 lseek(fd,0,SEEK_SET);
25
26 //读文件
27 char buf[1024];
28 memset(buf,0x00,sizeof(buf));
29 int n = read(fd,buf,sizeof(buf));
30 printf("n==[%d],buf==[%s]\n",n,buf);
31
32 //关闭文件
33 close(fd);
34
35 return 0;
36
37 }
~
使用lseek获取文件大小
//ff_t lseek(int fd, off_t offset, int whence)
//利用lseek获取文件大小
int len = lseek(fd,0,SEEK_END);
printf("file size :[%d]\n", len);
使用lseek实现文件拓展
//lseek函数实现文件拓展
//移动文件指针到第100个字节处
lseek(fd,100,SEEK_SET);
//必须紧接着一次写操作,否则不生效
write(fd,"A",1);
阻塞和非阻塞
通过读普通文件测试得知:read函数在读完文件内容之后,若再次read,则read函数会立刻返回,表明read函数读普通文件是非阻塞的。
设备文件:/dev/tty
通过读/dev/tty终端设备文件,表明read函数读设备文件是阻塞的。
结论:阻塞和非阻塞不是read函数的属性,而是文件本身的属性。