Linux 系统编程(IO)

工具

  • strace: 根据系统调用
  • od -tcx: 查看二进制

函数参数

  • 使用const修改的指针为传入参数
  • 不使用const的指针为传出参数

string操作的函数

  • strtok: 分隔字符串

IO

  • 函数

    • read
    • write
    • fcntl
    • iocntl
    • perror
    • strerror

IO-1

  • 头文件 -> unistd.h, fcntl.h, stdio.h, errno.h

  • open(fd, mode, permission) 函数

    • mode为宏-> O_RDONLY, O_CREAT, O_APPEND, O_TRUNC(会将文件截断, 将文件的大小改成0), O_RDWR, O_BLOCK, O_NONBLOCK, 当有O_CREAT时有第三个参数权限, 要考虑到umask, open函数使用可变参数实现了函数重载
  • read(fd, buf, sizeof(buf))

  • write(fd, buf, ret) // ret是read函数返回的字节数, 0表示达到了文件末尾, -1是异常

  • close(fd)

IO-2

阻塞

  • 在读取终端设备或者网络设备的时候会出现阻塞, 也就是说在读取这些设备的时候可以这些设备并没有数据可读, 程序就会阻塞在这里

  • 默认情况下, open("/dev/tty", O_RDONLY)是阻塞的, 也就是说只有终端有输入才会往下执行程序

  • open函数进行非阻塞读取, 需要借助宏O_NONBLOCK设置位

    • 示例(这里忽略里错误处理)


      int fd = open("/dev/tty", O_RDONLY | O_NONBLOCK);
      int ret = 0;
      char buf[1024];
      char *msg = "try again..."; while (ret = read(fd, buf, sizeof(buf))) {
      if (-1 == ret) {
      // 注意
      if (errno != EAGAIN) {
      // STDOUT_FILENO 是数字1, STDIN_FILENO 是数字0
      perror("error");
      exit(1);
      }
      write(STDOUT_FILENO, msg, strlen(msg));
      sleep(1);
      continue;
      }
      write(STDOUT_FILENO, buf, ret);
      }
    • 在上例中, 如果终端没有数据, 则read函数返回-1, 但是这并不代表着read错误了, 如果没有数据, Linux会将errno这是为EAGAIN, 如果errno不是EAGAIN, 则读取错误

  • 在执行一个程序的时候, 默认会PCB指向的数组指针中会在0, 1, 2下标对应的位置创建file struct, 代表着stdin, stdout, stderr, 所以我们使用终端的时候, 终端已经被打开了, 在上面的程序中, 我们要向实现非阻塞的功能, 需要打开一个已经打开的文件, 这个是不必要的, Linux提供了fcntl函数用于修改已经打开的文件的属性

    • 示例

      // F_GETFL是Get File Flag, 获取文件属性, 文件属性就是一个int类型, 采用bitmap的方式实现保存多个属性
      int ret = 0;
      int flag = fcntl(STDOUT_FILENO, F_GETFL);
      // 将对应的位置1, 也许O_NONBLOCK是000000000000000000001000000
      // |即使加法, 在fnctl内部判断flag是否有O_NONBLOCK属性, 是通过flag & O_NONBLOCK, 如果返回的是1则表示有O_NONBLOCK属性, 0则表示没有
      flag |= O_NONBLOCK;
      char buf[1024];
      char *msg = "try again..."; while (ret = read(STDOUT_FILENO, buf, sizeof(buf))) {
      if (-1 == ret) {
      // 注意
      if (errno != EAGAIN) {
      // STDOUT_FILENO 是数字1, STDIN_FILENO 是数字0
      perror("error");
      exit(1);
      }
      write(STDOUT_FILENO, msg, strlen(msg));
      continue;
      }
      write(STDOUT_FILENO, buf, ret);
      }
  • 移动文件指针(lseek)

    • 三个表示状态的宏: SEEK_SET(文件起始位置), SEEK_CUR(当前位置), SEEK_END(末尾)
    • lseek(fd, offset, macro)返回当前指针到其实位置的偏移量, 可以用来估计文件大小
    • 示例

      int fd = open("./test.txt", O_RDONLY | O_CREAT, 0644);
      // 文件大小
      int offset = lseek(fd, 0, SEEK_END);
      // 追加数据
      write(...);
      close(fd);
示例cp

int main(int argc, char *argv[]) {
char buf[1024] = {'\0'};
int ret = 0;
if (argc < 3) {
printf("Too Few Parameters\n");
return 1;
}
int fd_src = open(argv[1], O_RDONLY);
int fd_dst = open(argv[2], O_RDWR | O_CREAT | O_TRUNC, 0666);
while ((ret = read(fd_src, buf, sizeof(buf)))) {
write(fd_dst, buf, ret);
}
close(fd_src);
close(fd_dst);
return 0;
}

Linux IO的更多相关文章

  1. 【知乎网】Linux IO 多路复用 是什么意思?

    提问一: Linux IO多路复用有 epoll, poll, select,知道epoll性能比其他几者要好.也在网上查了一下这几者的区别,表示没有弄明白. IO多路复用是什么意思,在实际的应用中是 ...

  2. Linux IO模型和网络编程模型

    术语概念描述: IO有内存IO.网络IO和磁盘IO三种,通常我们说的IO指的是后两者. 阻塞和非阻塞,是函数/方法的实现方式,即在数据就绪之前是立刻返回还是等待. 以文件IO为例,一个IO读过程是文件 ...

  3. block_dump观察Linux IO写入的具体文件(mysqld)

      一.使用方法: 二.基本原理: 三.总结 很多情况下开发者调测程序需要在Linux下获取具体的IO的状况,目前常用的IO观察工具用vmstat和iostat,具体功能上说当然是iostat更胜一筹 ...

  4. linux io优化

    场景:xml文件解析入库:并备份 问题:磁盘io异常,经常100%busy: linux io优化方法: 1.修改磁盘挂着参数,修改为writeback模式:对于文件读取频繁的可以设置noatime: ...

  5. Linux IO 调度器

    Linux IO Scheduler(Linux IO 调度器) 每个块设备或者块设备的分区,都对应有自身的请求队列(request_queue),而每个请求队列都可以选择一个I/O调度器来协调所递交 ...

  6. Linux IO工具 iotop备择方案iopp

    iotop毫无疑问linux IO检测上是一个很好的工具,但苦于要求和内核版本Python版本号.我的很多朋友放弃了.我也是.无意中发现iopp,使用c书面,与此iotop它是一个作用.nice! 一 ...

  7. 【转】linux IO子系统和文件系统读写流程

    原文地址:linux IO子系统和文件系统读写流程 我们含有分析的,是基于2.6.32及其后的内核. 我们在linux上总是要保存数据,数据要么保存在文件系统里(如ext3),要么就保存在裸设备里.我 ...

  8. Linux IO时事检测工具iostat

    Linux IO时事检测工具iostat iostat命令用于检测linux系统io设备的负载情况,运行iostat将显示自上次运行该命令以后的统计信息.用户可以通过指定统计的次数和时间来获得所需的统 ...

  9. Linux IO Scheduler(Linux IO 调度器)【转】

    每个块设备或者块设备的分区,都对应有自身的请求队列(request_queue),而每个请求队列都可以选择一个I/O调度器来协调所递交的request.I/O调度器的基本目的是将请求按照它们对应在块设 ...

  10. linux io的cfq代码理解

    内核版本: 3.10内核. CFQ,即Completely Fair Queueing绝对公平调度器,原理是基于时间片的角度去保证公平,其实如果一台设备既有单队列,又有多队列,既有快速的NVME,又有 ...

随机推荐

  1. Vue 兄弟组件通过事件广播传递数据

    非父子组件传值 通过事件广播实现非父子组件传值1.新建js,引入并实例化Vue import Vue from 'vue' var VueEvent = new Vue(); export defau ...

  2. day08.4-samba共享网盘服务

    1. 安装软件:yum   install   samba   -y 2. 新建共享目录物理路径:mkdir   /zizaijiapu 修改配置文件:vim   /etc/samba/smb.con ...

  3. leecode刷题(1)-- 删除排序数组中的重复项

    删除排序数组中的重复项 给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度.不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的 ...

  4. 【hadoop】 eclipse中的“run on hadoop”和打包成jar提交任务的区别

    eclipse中的 调试运行 及 “run on hadoop”默认只是运行在单机上的,因为要想在集群中让程序分布式运行还要经历上传类文件.分发到各个节点等过程, 一个简单的“run on hadoo ...

  5. django 部署到Ubuntu安装MYSQL56

    阿里云 Ubuntu 14.04 安装mysql 5.6 1.升级apt-get sudo apt-get update 2. 安装mysql5.6版本 apt-get install mysql-s ...

  6. C++20草案中的宇宙飞船运算符(<=>,spaceship operator)

    C++20草案中的宇宙飞船运算符(<=>,spaceship operator) Herb Sutter提议的新三路运算符<=>已经被合入C++20草案中. 宇宙飞船运算符(h ...

  7. 二、为什么要选用pytest以及 pytest与unittest比较

    为什么要选择pytest,我看中的如下: 写case,不需要像unittest那样,创建测试类,继承unittest.TestCase pytest中的fixture(类似于setUp.tearDow ...

  8. js new关键字 和 this详解

    构造函数 ,是一种特殊的函数.主要用来在创建对象时初始化对象, 即为对象成员变量赋初始值,总与new运算符一起使用在创建对象的语句中. 构造函数用于创建一类对象,首字母要大写. 构造函数要和new一起 ...

  9. COCO2018 目标检测

    刚浏览了一下coco数据集官网,认真看了一下18年的目标检测任务,简单记录一下. coco2018目标检测挑战赛只进行实例分割的评比,虽然仍然可以输出bbox,但是不可以提交到比赛的服务器,原因是官方 ...

  10. hdu3966 Aragorn's Story 树链剖分

    题目传送门 题目大意: 有n个兵营形成一棵树,给出q次操作,每一次操作可以使两个兵营之间的所有兵营的人数增加或者减少同一个数目,每次查询输出某一个兵营的人数. 思路: 树链剖分模板题,讲一下树链剖分过 ...