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. javascript framework js常用框架

    js常用框架 一.node.js   二.angularjs.js   三.react.js   四.webpack.js   五.flux.js   六.vue.js   七.bootstrap   ...

  2. 为openstack服务使能debug模式

    Most OpenStack services use the same configuration options to enable the debug logging that is also ...

  3. Socket 简易静态服务器 WPF MVVM模式(三)

    ViewModels类 这个类其实就是个大杂烩,什么都可以用 这个类没有什么东西,最多的就是写一下xaml页面的逻辑控制,开启关闭监听 using System; using System.Colle ...

  4. React基础篇 (1)-- render&components

    render 基础用法 //1.创建虚拟DOM元素对象 var vDom=<h1>hello wold!</h1> //2.渲染 ReactDOM.render(vDom,do ...

  5. IIS发布的网页上传文件被拒绝

    在IIS所在的服务器共享的权限(如下图示,但注意不是加everyone)和共享文件夹的权限里都加上IIS_USER完全控制,如果不行再加上NETWORK SERVICE权限

  6. ThinkCMF if 标签

    用法示例: <if condition="($name eq 1) OR ($name gt 100) "> value1 <elseif condition=& ...

  7. 没有上司的舞会 树形dp

    题目描述 某大学有N个职员,编号为1~N.他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司.现在有个周年庆宴会,宴会每邀请来一个职员都会增加一定的快乐指数Ri, ...

  8. C# 根据列名获取列值

    /// <summary> /// 根据列名获取列值 /// </summary> /// <param name="colName">< ...

  9. [BZOJ 5330][SDOI2018] 反回文串

    传送门 怎么说呢,一道不可多得的反演题吧,具体解释之后再补 #include <bits/stdc++.h> using namespace std; #define rep(i,a,b) ...

  10. CF E2 - Array and Segments (Hard version) (线段树)

    题意给定一个长度为n的序列,和m个区间.对一个区间的操作是:对整个区间的数-1可以选择任意个区间(可以为0个.每个区间最多被选择一次)进行操作后,要求最大化的序列极差(极差即最大值 - 最小值).ea ...