文件I/O(2)

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;
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值