Linux系统IO

news2025/1/4 21:55:04

文章目录

  • Linux系统IO
      • sysio简介
      • sysio版本的copy
          • 示例代码
          • 代码说明
          • 函数讲解
          • 如何编译 运行
      • 系统IO与标准IO的区别
          • 示例代码
          • 函数讲解
          • 编译 运行?
        • 程序中的重定向
          • 代码示例
          • 代码说明
          • 函数讲解
          • 编译 运行?
          • 代码示例
          • 函数讲解
          • 编译 运行

Linux系统IO

sysio简介

所谓文件IO就是指文件的 输入(input)和 输出(output)
Linux 把大部分系统资源当作文件并呈现给用户,用户只需按照文件 I/O 的方式,就能完成数据的输入输出。 Linux 文件按其代表的具体对象,可大致分类为:

  • 普通文件,即一般意义上的文件、磁盘文件;
  • 设备文件,代表的是系统中一个具体的设备;
  • 管道文件、 FIFO 文件,一种特殊文件,常用于进程间通信;
  • 套接字(socket)文件,主要用在网络通信方面。
    Linux 系统提供的文件 I/O 接口函数,是以最基本的系统服务形式提供的,又称它们为基本 I/O 函 数。这些函数有个共同的特点: 它们都通过文件描述符(file descriptor)来完成对指定文件的 I/O 操 作。文件描述符 fd(file descriptor)是进程中用来表示某个文件的整数, 有的文献资料中又称它为文件句柄(file handle)。
    在Linux中,文件IO有两种实现方式,一种称为标准IO(stdio),一种称为系统IO(sysio)。这两者也就类似于普通话与方言的区别。
    在本文来介绍系统IO,系统IO与标准IO不同的是:系统IO有一个概念叫文件描述符fd贯穿始终,而在标准IO中是FILE类型贯穿始终,我们知道系统IO是指系统底层的syscall(系统调用),每一个系统都有不同的系统调用,标准IO的出现才有了意义。
    接下来,请跟我着手几个小例子,一起掌握Linux下的系统IO吧。

sysio版本的copy

大家应该都在shell中使用过cp命令吧。
请添加图片描述
cp命令用来copy一个文件或者是文件夹,这次我们使用sysio实现一个简易版的cp命令。
对于Linux自带的cp来说,要求我们在终端中输入具体的选项以及原路径和目标路径。
我们的简易版的mycopy也要具备这样的功能。

示例代码
#include "iostream"
#include "stdio.h"
#include "sys/stat.h"
#include "sys/wait.h"
#include "sys/types.h"
#include "unistd.h"
#include "errno.h"
#include "string.h"
#include <fcntl.h>

using namespace std;

#define BUFSIZE 9

int main(int argc, char **argv)
{
    char buffer[BUFSIZE];
    ssize_t ret = 0;
    if (argc < 3)
    {
        fprintf(stderr, "Usage....\n");
        exit(1);
    }
    /*
    以只读的方式打开src文件
    */
    int sfd = open(argv[1], O_RDONLY);
    if (sfd < 0)
    {
        /*源文件打开失败*/
        fprintf(stderr, "open() %s\n", strerror(errno));
        exit(1);
    }
    int dfd = open(argv[2], O_WRONLY | O_CREAT | O_TRUNC, 0600);
    if (dfd < 0)
    {
        /*目标文件打开失败*/
        /*记得关闭源文件*/
        close(sfd);
        fprintf(stderr, "open() %s\n", strerror(errno));
        exit(1);
    }
    /*从源文件中读取BUFSIZE个字节放道数据缓冲区中*/
    int len = read(sfd, buffer, BUFSIZE);
    if (len <= 0)
    {
        fprintf(stderr, "read() %s\n", strerror(errno));
        exit(1);
    }
    if (len > 0)
    {
        /*以防写入不足*/
        ret = write(dfd, buffer, BUFSIZE);
        if (ret < 0)
        {
            fprintf(stderr, "write() %s\n", strerror(errno));
            exit(1);
        }
    }
    close(dfd);
    close(sfd);
    exit(0);
}
代码说明

以只读的方式打开源文件src,从src中读取BUFSIZE个字节到buffer缓冲区中。同时以只写的方式打开目标文件dest,从buffer缓冲区中向目标文件写入数据,一次性写入BUFSIZE个字节

函数讲解

open()函数定义如下:
open()函数打开或者创建一个文件
在这里插入图片描述

  int open(const char *pathname, int flags);
  int open(const char *pathname, int flags, mode_t mode);

open()函数要求传入一个const char *pathname路径名称。
参数flags参考如下:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
通过open()函数创建的文件权限是多少呢?man手册给出了明确的解释:
在这里插入图片描述
当前的创建的文件权限为mode & ~umask。
同时,open()函数会返回一个打开文件的文件描述符fd,如果出错则返回-1,并设置errno。
在这里插入图片描述
read()函数定义如下:
read()函数从一个fd文件描述符中读取数据
在这里插入图片描述

ssize_t read(int fd, void *buf, size_t count);
int fd:fd指向要读取的源文件
void *buf:存储源文件中读取的字符串
size_t count:读取的字节数

read()函数从fd对应的文件中读取字符串,一次性读取count个字节,读取的字符串会存放在缓冲区buffer中。

man手册中还提到了,如果操作成功,read()函数会返回读取的字节数;如果操作失败,会返回-1,并设置errno。
在这里插入图片描述
write()函数定义如下:
write()函数向fd对应的文件中写入数据。
在这里插入图片描述

ssize_t write(int fd, const void *buf, size_t count);
int fd:fd对应的目标文件
const void *buf:从缓冲区buffer中向fd对应的目标文件写入数据
size_t count:一次性向目标文件写入的字节数

在这里插入图片描述
如果成功,write()函数会返回写入的字节数,如果失败,会返回-1,并设置errno。
close()函数定义如下:
close()函数关闭已经打开的文件
在这里插入图片描述

int close(int fd);
int fd:已经通过open打开的文件对应的文件描述符

如果成功返回0,如果失败则返回-1,并设置errno。这个close()函数与标准IO中的fclose()函数一般都是
操作成功的,不需要进行异常处理

如何编译 运行

首先您的当前路径中要存在一个源文件,
加粗样式

g++ main.cpp -o mycopy
./mycopy src dest

在这里插入图片描述

系统IO与标准IO的区别

标准IO的吞吐量大 系统IO的响应速度快(是对程序而言,对用户而言stdio的速度“更快”)

示例代码
#include "iostream"
#include "stdio.h"
#include "stdlib.h"
#include "unistd.h"

int main(int argc, char **argv)
{
    for (int i = 0; i < 1025; i++)
    {
        putchar('a');
        write(1, "b", 1);
    }
    exit(0);
}
函数讲解

标准IO与系统IO联系的纽带:fileno() fdopen()
我们要注意,标准IO与系统IO不可以混用。
fileno()定义如下:
在这里插入图片描述

int fileno(FILE *stream);

fileno()函数要求输入一个FILE类型,并返回一个文件描述符
fdopen()定义如下:
在这里插入图片描述

FILE *fdopen(int fd, const char *mode);
int fd:已经打开文件的对应文件描述符
const char *mode:权限模式
在这里插入图片描述
将FILE流与现有的文件描述符相关联。
如果成功返回一个文件指针;如果失败返回一个NULL,并设置errno

编译 运行?
g++ main.cpp -o main.o
./main.o

在这里插入图片描述

strace可以用来查看一个可执行文件的系统调用,使用strace ./main.o可以看到先进行1024次系统调用然后缓冲区满了1024合并为一次系统调用
在这里插入图片描述
putchar()是标准IO write()是系统IO。标准IO的吞吐量大 系统IO的响应速度快(是对程序而言,对用户而言stdio的速度“更快”)
在考虑到源代码的移植性好,通常选用标准IO而不是系统IO

程序中的重定向

dup 将传入的文件描述符复制到可使用的(未使用的)最小新文件描述符,在下面的例子中,将标准输出关闭后,文件描述符1空闲(不发生竞争时),dup将会把打开了文件out的文件描述符复制给文件描述符1 ,之后对文件描述符1 的操作就相当与操作文件out

代码示例
#include "stdio.h"
#include "iostream"
#include "stdlib.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include "string.h"
#include "errno.h"
#include "unistd.h"

using namespace std;

#define FNAME "out"

int main(int argc, char **argv)
{
    int fd = -1;
    fd = open(FNAME, O_WRONLY | O_CREAT | O_TRUNC, 0600);
    if (fd < 0)
    {
        fprintf(stderr, "open() %s\n", strerror(errno));
        exit(1);
    }
    close(1);
    dup(fd);
    close(fd);
    printf("hello world\n");
    exit(0);
}
代码说明

通过open()函数以只写的方式打开FNAME文件,如果不存在这个文件则进行创建,如果存在则执行截断操作,并赋予对应的权限。如果fd < 0 代表错误,输出对应的错误信息并退出。如果成功,关闭1号文件描述符,使用dup()函数复制fd。

函数讲解

dup定义如下:
dup()函数是复制一个文件描述符
在这里插入图片描述

int dup(int oldfd);
int oldfd:一个文件描述符fd

dup()系统调用创建文件描述符oldfd的副本,为新的描述符使用编号最低的未使用文件描述符。
在这里插入图片描述
dup()函数如果操作成功,会返回一个新的文件描述符,如果出错返回-1,并设置errno。
注意:dup()函数的操作不原子
何谓原子性操作,即为最小的操作单元,比如i=1,就是一个原子性操作,这个过程只涉及一个赋值操作。 又如i++就不是一个原子操作,它相当于语句i=i+1;这里包括读取i,i+1,结果写入内存三个操作单元。 因此如果操作不符合原子性操作,那么整个语句的执行就会出现混乱,导致出现错误的结果,从而导致线程安全问题。

编译 运行?

在这里插入图片描述
由于dup()操作不原子,会导致线程安全问题,接下来会介绍一个具有原子性操作的函数dup2()

代码示例
#include "stdio.h"
#include "stdlib.h"
#include "iostream"
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include "string.h"
#include "errno.h"
#include "unistd.h"

using namespace std;

#define FNAME "out"

int main(int argc, char **argv)
{
    int fd = -1;
    fd = open(FNAME, O_WRONLY | O_CREAT | O_TRUNC, 0600);
    if (fd < 0)
    {
        fprintf(stderr, "open() %s\n", strerror(errno));
        exit(1);
    }
    dup2(fd,1);
    if (fd != 1)
    {
        close(fd);
    }
    printf("hello world\n");
    exit(0);
}
函数讲解

dup2()函数:
dup2()函数可以避免关闭文件描述符后被其他线程抢占
在这里插入图片描述

int dup2(int oldfd, int newfd);

dup2()和dup()函数一样,同样都是复制文件描述符,但它不使用编号最低的未使用文件描述符,它使用newfd中指定的文件描述符编号。
dup()函数在使用是,是返回的编号最低的未使用的文件描述符,这个文件描述符极可能会被其他文件占用,而dup2()函数就恰好解决了这个问题。
同样,dup2()函数成功会返回新的文件描述符,如果失败则返回-1,并设置errno。

编译 运行

在这里插入图片描述

同步

sync 设备即将解除挂载时进行全局催促,将buffer cache的数据刷新
fsync 刷新文件的数据
fdatasync 刷新文件的数据部分,不修改文件元数据

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/75630.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

全国青少年软件编程(Scratch)等级考试二级考试真题2022年6月——持续更新.....

电子学会202206Scratch二级真题及参考答案 1.角色初始位置如图所示&#xff0c;下面哪个选项能让角色移到舞台的左下角&#xff1f;&#xff08; &#xff09; A. B. C. D. 正确答案&#xff1a;C 答案解析&#xff1a; 舞台的左下角&#xff0c;坐标x为负数&#xff0c;…

【数据结构】七大排序

目录 一、什么是稳定性 二、七大排序 2.1基于选择的思想 2.1.1直接选择排序 2.1.2堆排序 2.2基于插入的思想 2.2.1直接插入排序 2.2.2希尔排序 2.3归并排序 2.4基于交换的思想 2.4.1冒泡排序 2.4.2快速排序 三、外部排序 排序就是将一组无序的数据经过一定的算法调…

Linux系统:root用户 登录失败

问题 在Linux系统上&#xff0c;从root用户切换到oracle用户时报错 su: cannot open session: Permission denied 如下&#xff1a; 分析 定位原因1 分析登录日志&#xff0c;可以看到时登录的时候limit中的 memlock 设置失败&#xff0c;导致用户登录失败&#xff1a; limi…

[GO] Gin入门

1. Gin基本使用 1.1 Gin入门 Gin是一个golang的微框架,封装比较优雅,API友好,源码注释比较明确,具有快速灵活,容错方便等特点对于Golang而言,web框架的依赖要远比Python,Java之类要小,自身的net/http足够简单,性能也非常不错借助框架开发,不仅可以省去很多常用的封装带来的时…

为本地web服务配置使用固定的二级子域名【内网穿透】

由于使用免费的cpolar所生成的公网地址为随机临时地址&#xff0c;该地址24小时内会发生变化&#xff0c;对于需要长期访问的用户来讲比较不方便。 不过我们可以为其配置cpolar固定的二级子域名&#xff08;该二级子域名可自定义&#xff09;&#xff0c;该地址不会随机变化&a…

芯片漫游指南(2)-- UVM结构

目录&#xff1a;1 组件家族1.1概述1.2 uvm_driver1.2.1 概述1.2.2 示例1.3 uvm_monitor1.3.1 概述1.3.2 示例1.4 uvm_sequencer1.4.1 概述1.4.2 示例1.5 uvm_agent1.5.1 概述1.5.2 示例1.6 uvm_scoreboard1.6.1 概述1.6.2 示例1.7 uvm_env1.7.1 概述1.7.2 示例1.8 uvm_test1.8…

天翎携手群晖助力电商行业文档管理

编者按&#xff1a;电商行业的文档管理怎么做&#xff1f;本文根据电商行业文档管理中存在的一些难点&#xff0c;提出天翎文档管理系统和群晖NAS结合的解决方案。 关键词&#xff1a;免安装&#xff0c;免维护&#xff0c;文件分类&#xff0c;权限设置&#xff0c;文件同步&…

【拿捏链表(Ⅱ)】—Leetcode删除排序链表中的重复元素

目录删除排序链表中的重复元素(Ⅰ)删除排序链表中的重复元素(Ⅱ)删除排序链表中的重复元素(Ⅰ) 题目&#xff1a; 给定一个已排序的链表的头 head &#xff0c;删除所有重复的元素&#xff0c;使每个元素只出现一次 。返回 已排序的链表 。 思路&#xff1a;这里的思路很简单&…

Navigation--导航算法(局部视野导航)--DWA、TAB

DWA 动态窗口法&#xff08;dynamic window approach&#xff0c;DWA&#xff09;目前与A*一样都是ROS导航包中提供的基本路径规划算法。DWA是一种贪心的算法&#xff0c;通过可选速度、可选角速度的组合&#xff0c;模拟出很多局部轨迹&#xff0c;然后选择最优的。这种方法时…

Android dex动态加载(Kotlin版)

前言 环境 语言–KotlinJDK11SDK33AndroidStudio版本 Android Studio Dolphin | 2021.3.1 Patch 1 Build #AI-213.7172.25.2113.9123335, built on September 30, 2022概述 libaray项目打包成jarjar通过dx命令行工具转为dex.jardex.jar放到assets目录下App启动读取assets中…

外贸业务12年,我想和大家分享这几点感受

如今再回看这段经历&#xff0c;很庆幸我的三观一直都很正确&#xff0c;那就是买家第一。 不管是什么原因&#xff0c;只要你想退&#xff0c;我都可以接受退&#xff0c;我不能退回上级供应商的那我就自己留着&#xff0c;只为了不想因为这一次拒绝而失去这个买家&#xff1…

springboot集成security(鉴权)

本文承接上一章节内容&#xff1a;springboot集成security&#xff08;认证&#xff09; 上一章节&#xff1a; https://blog.csdn.net/m0_54355172/article/details/128239128 1. 授予静态资源访问权限 因为我的演示案例涉及到多个页面&#xff0c;所以先说一下如何给静态资源…

数据挖掘——关联规则(Association Rule)Apriori算法和python代码实现

关联规则&#xff08;Association Rule&#xff09;什么是关联规则一些基本概念任务是什么Apriori 算法核心思想步骤与流程图如何找到候选集python代码实现什么是关联规则 关联规则(Association Rules)是反映一个事物与其他事物之间的相互依存性和关联性&#xff0c;是数据挖掘…

线程2的深度剖析

加锁 synchronized 1.修饰方法&#xff08;普通方法&#xff0c;静态方法&#xff09;普通方法实际上加到了this上&#xff0c;静态方法加到了类对象上。 2.修饰代码块 手动指定加到那个对象上 明确锁对象针对那个对象加锁&#xff0c;如果两个线程针对同一个对象加锁&am…

(1)AWD入门攻略大纲

1.比赛介绍 (1)比赛环境 (2)常见服务器信息介绍 比赛名称 白名单&#xff1a;一般用于防止外部恶意攻击&#xff0c;如果赛方发现名单以外IP可能会进行封禁处理。 服务器账号密码 Token和虚拟IP&#xff1a;token为提交答案的凭证&#xff0c;绑定了队伍&#xff1b;虚拟IP为…

SpringCloud中Feign注解@FeignClient参数一览表

写在前面 Feign是微服务中服务间调用的优选组件&#xff0c;后来的OpenFeign也是基于此来开展的。 为什么要梳理一下Feign注解FeignClient中的各个参数&#xff1f; 踩坑太多面试总问 参数一栏表 FeignClient的源码示例图如下&#xff1a; 今天我们接着来说最后的几个参数。…

Java面试题(六)美团JVM夺命7连问(灵魂拷问)

0.来看一道美团的面试题 这题直接把人给问懵逼了&#xff0c;你能全部答出来吗&#xff1f; Object o new Object();请解释对象的创建过程&#xff1f;DCL要不要加volatile问题&#xff1f;对象在内存中的存储布局&#xff1f;什么是指针压缩&#xff1f;对象头具体包含哪些…

生成树问题汇总

生成树问题汇总注1、最小(大)生成树思路代码例子&#xff1a;1、最小生成树结果是2、最大生成树结果2、在最小生成树中再加一条边&#xff0c;求新的最小生成树思路代码核心代码全部代码例子3、次小生成树思路:在上一个功能基础上进一步扩充代码核心代码全部代码例子4、判断最小…

一个轻量级的分布式日志标记追踪神器,十分钟接入,非常好用!

TLog简介 1、TLog通过对日志打标签完成企业级微服务的日志追踪。它不收集日志&#xff0c;使用简单&#xff0c; 产生全局唯一的追踪码。除了追踪码以外&#xff0c;TLog还支持SpanId和上下游服务信息 标签的追加。 2、为用户使用方便而设计&#xff0c;提供完全零侵入式接入…

es入门(上)

笔记来源于学习 b站中的【IT李老师】的elasticsearch课程 自己在实习做的es模块中的理解。 后续会有 中&#xff0c;下篇笔记更新&#xff0c;目前这一篇是上篇。 目录 Elastic Stack简介 1.1简介 1.2特色 1.3组件介绍 2.Elasticsearch的接收与核心概念 2.1搜索是什么…