1.PCB
PCB(process control block)进程控制块,操作系统把每一个进程当一个PCB去管理
进程控制块其实是一个结构体
cd /
find -name sched.h
vi ./usr/src/linux-headers-5.13.0-37/include/linux/sched.h
在里面找到task_struct的结构体
靠结构体成员去描述进程的状态
我们知道FILE*中有对应的文件描述符,文件描述符和真正的设备文件发生关联
其整个过程为:
在file_struct找到对应的文件描述符
file_struct结构体指向文件描述符
文件描述符表,可以先简单认为是一个整型数组
2.open/close
open函数可以打开或创建一个文件。
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
int open(const char *pathname, int flags);
int open(const char *pathname, int flags, mode_t mode);
返回值:成功返回新分配的文件描述符,出错返回-1并设置errno
flags
必选项:以下三个常数中必须指定一个,且仅允许指定一个。
* O_RDONLY 只读打开
* O_WRONLY 只写打开
* O_RDWR 可读可写打开
以下可选项可以同时指定0个或多个,和必选项按位或起来作为flags参数。可选项有很多,
这里只介绍一部分,其它选项可参考open(2)的Man Page:
* O_APPEND 表示追加。如果文件已有内容,这次打开文件所写的数据附加到文件的末尾
而不覆盖原来的内容。
* O_CREAT 若此文件不存在则创建它。使用此选项时需要提供第三个参数mode,表示该
文件的访问权限。
* O_EXCL 如果同时指定了O_CREAT,并且文件已存在,则出错返回。
* O_TRUNC 如果文件已存在,并且以只写或可读可写方式打开,则将其长度截断(Trun-
cate)为0字节。
* O_NONBLOCK 对于设备文件,以O_NONBLOCK方式打开可以做非阻塞I/O(Nonblock I/
O),非阻塞I/O在下一节详细讲解。这就是open函数比fopen强大的原因
返回值:返回文件描述符,若出错,返回-1。
3.小实例
(1)
#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
//一个open要 三个头文件
int main(){
int fd;
fd=open("abc",O_CREAT,0777);
printf("fd=%d\n",fd);
return 0;
}
设置权限为777,于umask按位与之后的结果是775
(2)创建一个文件,自己设置文件名
#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
//一个open要 三个头文件
#include<stdlib.h>
#include<string.h>
#include<unistd.h>//unix标准头文件
//close
int main(int argc,char *argv[]){
if(argc<2){
printf("please ./app + filename");
exit(1);
}
int fd;
char buf[]="helloworld";
fd=open(argv[1],O_CREAT|O_RDWR,0777);
write(fd,buf,strlen(buf)); //sizeof strlen
printf("fd=%d\n",fd);
close(fd);
return 0;
}
* exit和return的区别:在任何函数中调exit都会程序结束
* open一个文件之后记得close,close函数成功返回0,失败返回-1
注意:
open函数与C标准I/O库的fopen函数有些细微的区别:
(1)以可写的方式fopen一个文件时,如果文件不存在会自动创建,而open一个文件时必须
明确指定O_CREAT才会创建文件,否则文件不存在就出错返回。
(2)以w或w+方式fopen一个文件时,如果文件已存在就截断为0字节,而open一个文件时必须明确指定O_TRUNC才会截断文件,否则直接在原来的数据上改写。
(3)第三个参数mode指定文件权限,可以用八进制数表示,比如0644表示-rw-r-r–,也可
以用S_IRUSR、S_IWUSR等宏定义按位或起来表示,详见open(2)的Man Page。要注意的是,文件权限由open的mode参数和当前进程的umask掩码共同决定。
* 最大打开文件个数,默认为1024个
open files 为1024,可以修改。
4.read/write
read函数:
从打开的设备或文件中读取数据。
#include <unistd.h>
ssize_t read(int fd, void *buf, size_t count);
返回值:成功返回读取的字节数,出错返回-1并设置errno,如果在调read之前已到达文件末尾,则这次read返回0
write函数
向打开的设备或文件中写数据。
#include <unistd.h>
ssize_t write(int fd, const void *buf, size_t count);
返回值:成功返回写入的字节数,出错返回-1并设置errno
5.完成一个mycopy
#include<stdio.h>
#include<sys/types.h>
#include<fcntl.h>
#include<sys/types.h>
#include<fcntl.h>
#include<unistd.h>
#include<stdlib.h>
#include<string.h>
#define SIZE 8192
int main(int argc,char *argv[]){
char buf[SIZE];
int fd_src,fd_dest,len;
if(argc<3){
printf("./mycopy src dest\n");
exit(1);
}
fd_src=open(argv[1],O_RDONLY);
fd_dest=open(argv[2],O_CREAT|O_WRONLY|O_TRUNC,0644);
while(len=read(fd_src,buf,sizeof(buf))>0){
write(fd_dest,buf,len);
}
//read读失败返回-1,读到文件末尾返回0
//成共返回读到的字节数
close(fd_src);
close(fd_dest);
return 0;
}