read/write函数
- ssize_t read(int fd, void *buf, size_t count);
- ssize_t write(int fd,const void *buf,size_t count);
read和write函数原型类似,要注意第三个参数
write()
//头文件:#include<unistd.h>
原型:
ssize_t write(int fd,const void*buf,size_t count);
参数说明:
fd:是文件描述符(write所对应的是写,即就是1)
buf:通常是一个字符串,需要写入的字符串
count:是每次写入的字节数
返回值:
成功:返回写入的字节数
失败:返回-1并设置errno
ps: 写常规文件时,write的返回值通常等于请求写的字节
数count, 而向终端设备或者网络写时则不一定
read()
头文件:#include<unistd.h>
功能:用于从文件描述符对应的文件读取数据(从打开的设备或文件中读取数据)
原型:
ssize_t read(int fd,void*buf,size_t count)
参数说明:
fd: 是文件描述符
buf: 为读出数据的缓冲区;
count: 为每次读取的字节数(是请求读取的字节数,读上来的数据保
存在缓冲区buf中,同时文件的当前读写位置向后移)
返回值:
成功:返回读出的字节数
失败:返回-1,并设置errno,如果在调用read
之前到达文件末尾,则这次read返回0
如果为-1,并且errno = EAGIN或者EWOULBLOCK,说明不是read失败,而是read以非阻塞方式读一个设备或者网络文件,并且文件无数据
代码:
#include<stdio.h>
#include<string.h>
#include<unistd.h>
int main()
{
const char*msg="hello\n";
int len = strlen(msg);
write(1,msg,len);//write所对应的文件描述符为1
char buf[1024]={0};
read(0,buf,len);//read所对应的文件描述符为0
return 0;
}
fread函数read函数的区别
- fread函数是封装好的库函数,而read函数是系统函数,一般来说,fread效率更高;
- 读取文件的差别:fread函数功能更强大,可以读取结构体的二进制文件,但是如果是最底层的操作,用到文件描述符的话,用read会更好。
文件描述符
- PCB进程控制块:本质就是结构体
- 成员:文件描述符表
- 文件描述符:0/1/2…/1023,表中可用最小的
理解就是0-4G的虚地址空间里的PCB进程控制块,里面有一个文件描述符表,存放的的文件结构体指针,文件描述符就是这个文件描述符的下标,其中0 1 2一般不可用,从3开始,最大到1023,也就是一个进程最多打开1024个文件。
阻塞和非阻塞
- 常规文件不会阻塞
- 读网络和读设备这两种情况会阻塞
- 阻塞是文件的属性
fcntl
修改文件状态函数
int flags = fcntl(fd, F_GETFL);
flag |= O_NONBLOCK;
fcntl(fd, F_SETFL,flags);
//获取文件状态:F_GETFL
//设置文件状态:F_SETFL