Linux:操作文件的系统调用


需要引入的头文件:

#inlcude<unistd.h>

1.打开文件

  1. 打开一个已存在的文件
int open(const char *pathname, int flags);
  1. 新建一个文件并创建权限
int open(const char *pathname, int flags, mode_t mode);

参数介绍

  1. pathname:将要打开的文件路径和名称
  2. flags:打开标志

标志介绍:

       The  argument  flags  must  include  one of the following access modes:
       O_RDONLY, O_WRONLY, or O_RDWR.  These request opening  the  file  read-
       only, write-only, or read/write, respectively.

  • O_RDONLY 只读打开
  • O_RDWR 读写打开
  • O_CREAT 文件不存在则创建
  • O_APPEND 文件末尾追加
  • O_TRUNC 清空文件,重新写入
  1. mode
The following symbolic constants are provided for mode:

S_IRWXU  00700 user (file owner) has read,  write,  and  execute permission
                       

S_IRUSR  00400 user has read permission

S_IWUSR  00200 user has write permission

S_IXUSR  00100 user has execute permission

S_IRWXG  00070 group has read, write, and execute permission

S_IRGRP  00040 group has read permission

S_IWGRP  00020 group has write permission

S_IXGRP  00010 group has execute permission

S_IRWXO  00007 others have read, write, and execute permission

S_IROTH  00004 others have read permission

S_IWOTH  00002 others have write permission

S_IXOTH  00001 others have execute permission


  1. 返回值:文件描述符

2. 读文件

ssize_t read(int fd, void *buf, size_t count);

参数介绍

  • fd:对应打开的文件描述符
  • buf : 存放数据的空间
  • count: 计划一次从文件中读多少字节数据
  • 返回值: 实际读到的字节数

3. 写文件

ssize_t write(int fd, const void *buf, size_t count);

参数介绍:

  • fd :对应打开的文件描述符
  • buf:存放待写入的数据
  • count:计划一次向文件中写入多少数据

4.关闭

int close(int fd);

  • fd :对应的文件描述符

分析题

如果父进程先打开一个文件,fork 后子进程是否可以共享使用?

文件内容
在这里插入图片描述

代码

#include<stdio.h>
#include<unistd.h>
#include<assert.h>
#include<fcntl.h>
#include<stdlib.h>

int main()
{
    char buff[128] = {0};

    int fd = open("myfile.txt", O_RDONLY);

    pid_t pid = fork();
    assert(pid != -1);

    if (pid == 0)
    {
        read(fd, buff, 1);
        printf("child buff = %s\n", buff);

        sleep(1);
        read(fd, buff, 1);
        printf("child buff = %s\n", buff);

    }
    else
    {
        read(fd, buff, 1);
        printf("parent buff = %s\n", buff);

        sleep(1);
        read(fd, buff, 1);
        printf("parent buff = %s\n", buff);
    }

    close(fd);

    exit(0);
}

运行结果:

在这里插入图片描述

结论
由于 fork 创建的子进程的 PCB 是拷贝父进程的,子进程的 PCB 中的文件表指向打开文件的指针只是拷贝了父进程 PCB 中的值,所以父子进程共享父进程 fork 之前打开的所有文件描述符。

在这里插入图片描述

练习题

完成对一个文件的复制(类似命令:cp)

原文件内容为:
在这里插入图片描述

代码:

#include<stdio.h>
#include<unistd.h>
#include<fcntl.h>
#include<stdlib.h>
#include<assert.h>

int main(void)
{
    char buff[128] = {0};

    int fdr = open("myfile.txt", O_RDONLY);
    assert(fdr != -1);

    int fdw = open("newfile.txt", O_WRONLY | O_CREAT, 0600);
    assert(fdw != -1);

    int n = 0;
    while (n = read(fdr, buff, 128) > 0)
    {
        write(fdw, buff, n);
    }

    close(fdr);
    close(fdw);
    
    exit(0);
}

运行示例:
可以看到newfile.txt创建成功

在这里插入图片描述

系统调用和库函数的区别

区别: 系统调用的实现在内核中,属于内核空间,库函数的实现在函数库中,属于用户空间。

系统调用执行过程:
在这里插入图片描述

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

_索伦

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值