linux学习笔记之IO

时间:2021-01-31 10:51:41

一、基础知识。

1:普通IO类型。

  1,非阻塞IO:发出open/read/write等IO操作,并使这些操作不会永远阻塞。当不能完成时,会立即出错返回。

    1)非阻塞的两种标志方式:指定标志:O_NONBLOCK。

    2)非阻塞语义:文件状态标志的更改影响同一文件表项的所有用户,但与通过其他文件表项对同一设备的访问无关。(关联于文件表项)

  2,异步IO(asynchronous IO)。

    1)机制:当描述符准备号可以进程IO时,发送一个信号通知进程。

      1-仅当描述符引用中断设备或网络时,它才能起作用。

      2-这种信号对每个进程都只有一个。

  3:IO多路转接(IO multiplexing)。

    1)机制:构造一个包含描述符的列表,然后调用一个函数。当描述符中的一个准备好进程IO操作时,该函数才返回。

    2)系统并不主动告知任何信息。需要我们使用函数取查询文件描述符。

  4:POSIX异步IO:为不同类型的文件进行异步IO提供例一套一致的方法。

    1)异步IO操作必须显式的指定偏移量。异步IO接口并不影响由操作系统维护的文件偏移量。

    2)使用追加模式时,aio_offset字段会被系统忽略。

  5:储存映射IO。

    1)机制:将一个磁盘文件映射到储存空间中的一段缓冲区上。

2:终端IO。

  1,两种工作模式。

    1)规范模式输入处理:按行处理,以行为结束。默认为规范模式。

    2)非规范模式输入处理:按各自的规则处理。

    3)规范模式下,有下列几种情况会造成读返回:请求字节数已读到,接收到行定界符,捕捉到信号。

    4)非规范模式的四种情况。TIME和MIN是termios结构中c_cc数组的两个变量。

\ MIN > 0 MIN == 0
TIME > 0

A:在定时器超时前,

  read返回[MIN, nbytes];

  如果定时器超时,

  read返回[1, MIN];

  (TIME=字节间定时器。调用者无限期阻塞。)

C:在定时器超时前,

  read返回[1, nbytes];

  如果定时器超时,

  read返回0.

  (TIME=read定时器)

TIME == 0

B:当有可用数据时,

  read返回:[MIN, nbytes]

  (调用者可无限期阻塞)

D:read立即返回[0, nbytes]

  2,POSIX.1 定义了11个特殊输入字符,其中9个可以更改。

  3,终端设备是由通常位于内核中的终端驱动程序控制的,每个终端设备都由一个输入队列和一个输出队列。

    1)输入队列填满时,系统性为依赖实现。

    2)输出队列填满时,进程通常休眠。直到队列有可用空间。

  4,终端IO函数汇总。

函数 说明
tcgetattr 获取属性
tcsetattr 设置属性
cfgetispeed 获取输入速度
cfgetospeed 获取输出速度
cfsetispeed 设置输入速度
cfsetospeed 设置输出速度
tcdrain 等到所有输出都被传输
tcflow 挂起 传输或接收
tcflush 冲洗未决输入和/或输出
tcsendbreak 发送BREAK字符
tcgetpgrp 获得前台进程组ID
tcsetpgrp 设置前台进程组ID
tcgetsid 得到控制TTY的会话首进程的进程组ID

  5,波特率:指位/秒(bit per second)。

    1)一般系统都定义了协议以外的波特率。

  6,伪终端。

    1)含义:指对于一个应用程序而言,它看上去像一个终端。但事实上并不是一个真正的终端。

    2)几个经典的用途P581(未完全了解。)。

      1-网络登录服务器。

      2-窗口系统终端模拟。

      3-script程序。

      4-expect程序。

      5-运行协同进程。

      6-观看长时间运行的程序的输出。

3:IO相关:记录锁功能

  1,记录锁功能(字节范围锁):一个进程正在读/修改文件的某个部分时,使用记录锁可以阻止其他进程修改同一个文件区。

    1)它可以只锁住文件的一段区域。控制上更为精确。

    2)共享读和独占写的控制模式和读写锁类似。

    3)单个进程在同一区域设置第二把锁时,会覆盖之前的锁。

    4)加读锁时,描述符必须时读打开;加写锁时,必须是写打开。

  2,锁的隐含继承和释放。

    1)当一个进程终止时,它建立的锁全部释放。当一个描述符关闭时,进程通过描述符设置的锁也会释放。

    2)由fork产生的子进程不继承父进程设置的锁。

    3)执行exec后,新程序可以继承原执行程序的锁(和上条似乎存在矛盾)。

  3,合作进程:一个库的所有函数都以一致的方法处理记录锁,则称使用这些函数访问数据库的进程集为合作进程。

  4,强制性锁:该锁会让内核见车每一个IO函数,验证调用进程是否违背了正在访问的文件上的某一把锁。

    1)打开方式:对特定文件打开其设置组ID位,关闭组执行位(无法理解)。

    2)建议锁的含义?

    3)强制性锁存在缺陷,是可以避开的。

4:其他。

  1,轮询:一段时间,调用一次期望的进程/函数。

    1)多任务系统中,尽量避免使用此方法。

  2,BSD派生系统中,异步IO是信号SIGIO 和 SIGURG 的组合。

    1)SIGIO:通用异步IO信号。

    2)SIGURG:通知进程网络连接上的带外数据已经到达。

二、相关函数。

1:多路转接IO

<sys/select.h>
1 多路转接的查询函数。
  int select( int maxfdp1, fd_set *restrict readfds, fd_set *restrict writefds, fd_set *restrict exceptfds, struct timeval *restrict tvptr );  
  // 1 输入:描述符,描述符条件(读/写/异常),等待时间。
  // 2 输出:描述符总量,已准备好的描述符条件
  // 3 参数tvptr: ==NULL 永远等待, ==0 不等待, !=0 等待具体时间。无const,无法保证不被修改。。
  // 4 fd_set类型参数(写/读/异常条件): 当参数==NULL时,表示不关心此参数。
  // 5 参数maxfdp1:最大文件描述符编号+1.
  // 6 返回值:出错-1,0 表示没准备好的描述符, >0 已准备好的描述符数之和。
  int pselect( int maxfdp1, fd_set *restrict readfds, fd_set *restrict writefds, fd_set *restrict exceptfds, const struct timespec *restrict tsptr, const sigset_t *restrict sigmask );
  // 1 提供高精度,并超时值无法改变。
  // 2 可使用 信号屏蔽字。
  int poll( struct pollfd fdarray[], nfds_t nfds, int timeout );
  // 1 将我们感兴趣的描述符写进 pollfd 数组中。
  // 2 参数 nfds:指定数组元素个数。
  // 3 参数timeout:-1 永远等待,0 不等待,>0 等待具体时间。
2 数据类型 fd_set 的处理函数/宏(取决于实现成函数,还是宏)。
  int FD_ISSET( int fd, fd_set *fdset ); // 指定位是否已打开。
  void FD_CLR( int fd, fd_set *fdset ); // 清除某一个位。
  void FD_SET( int fd, fd_set *fdset ); // 设置某一个位。
  void FD_ZERO( fd_set *fdset ); // 所有位设置为0。
  // 1 fd_set类型中,一个文件描述符占据一位。通常来说 位上置1 表示条件满足(不确定具体系统实现)。
3 数据类型 pollfd。
  struct pollfd
  {
    int fd; // file decriptor to check, or <0 to ignore.
    short events; // events of interest on fd.
    short revents; // events that occurred on fd.
  }
  // 1 如果使用此结构,需要直到具体的event返回值的含义

2:POSIX异步IO

1 AIO控制块基本结构(具体系统可在标准上添加)
  struct aiocb
  {
    int aio_fildes; // file descriptor
    off_t aio_offset; // file offset for IO
    volatile void *aio_buf; // buffer for IO
    size_t aio_nbytes; // number of bytes to transfer
    int aio_reqprio; // priority
    struct sigevent aio_sigevent; // signal information
    int aio_opcode; // operation for list IO
  }
2 基本读写操作。
  int aio_read( struct aiocb * aiocb );
  int aio_write( struct aiocb * aiocb );
3 强制等待的异步操作直接写入。
  int aio_fsync( int op, struct aiocb *aiocb ); // !!!具体功能未完全确认!!!
  // 1 op参数有两个选择:O_DSYNC---fdatasync, O_SYNC---fsync
4 获取一个操作完成状态。
  int aio_error( const struct aiocb *aiocb );
  // 1 返回值:0 成功,-1失败 errno中保存错误信息,EINPROGRESS 操作等待中,
5 如果成功,获取异步操作返回值。
  ssize_t aio_return( const struct aiocb *aiocb );
  // 1 调用一次后,系统就清除返回值。
6 进程只剩异步操作未完成,可以通过此函数阻塞进程。直到异步操作完成。
  int aio_suspend( const struct aiocb *const list[], int nent, const struct timespec *timeout );
7 取消等待的异步操作。
  int aio_cancel( int fd, struct aiocb *aiocb ); // 仅发出取消命令,并不代表一定取消。
  // 1 返回值:AIO_ALLDONE 所有操作已完成,不需要取消, AIO_CANCELED 成功, AIO_NOTCANCELED 最少有一个操作没被取消, -1 调用失败,错误保存在errno
8 提交一个 AIO控制块的 列表
  int lio_listio( int mode, struct aiocb *restrict const list[ restrict ], int nent, struct sigevent *restrict sigev );
  // 1 参数mode:LIO_WAIT, LIO_NOWAIT

3:储存映射IO。

1 将给定文件映射到储存区域。
  void *nmap( void *addr, size_t len, int prot, int flag, int fd, off_t off );
  // 1 参数addr:指定映射区域的起始地址。设置为0得到最大的可移植性。
  // 2 参数prot:储存区的权限(读/写/执行)
  // 3 参数flag:MAP_FIXED,MAP_SHARED,MAP_PRIVATE.
2 更改映射权限。
  int mprotect( void *addr, size_t len, int prot );
3 将储存区数据写入被映射文件。
  int msync( void *addr, size_t len, int flags );
  // 1 参数flags:MS_ASYNC, MS_SYNC
4 解除映射区。
  int munmap( void *addr, size_t len ); // 调用此函数,不会使储存区数据写入文件

4:多种读写函数。

1 读/写 多个缓冲区
  ssize_t readv( int fd, const struct iovec *iov, int iovcnt );
  ssize_t writev( int fd, const struct iovec *iov, int iovcnt );
  // 1 参数iovcnt:读取缓冲区的个数。
2 按照需求 读/写 N个字节的数据
  ssize_t readn( int fd, void *buf, size_t nbytes );
  ssize_t writen( int fd, void *buf, size_t nbytes );

5:记录锁

int fcnt1(int fd, int cmd, struct flock *flockptr );

struct flock
{
  short l_type; // F_RDLCK(共享读锁), F_WRLCK(独占写锁), F_UNLCK(解锁一个区域)
  short l_whence; // 加/解锁区域的起始字节偏移量。可选值:SEEK_SET, SEEK_CUR, SEEK_END
  off_t l_start; // 加/解锁区域的起始字节偏移量。
  off_t l_len; // 区域字节长度.等于0时,表示可以添加为最大偏移量,而不是0。
  pid_t l_pid; // 当前进程ID(cmd == F_GETLK 时)
}
// 1 对于记录锁,参数cmd是: F_GETLK(获取锁状态), F_SETLK(设置锁), F_SETLKW(设置锁的阻塞版,W==wait)。
// 2 参数flockptr时记录锁的一个结构体。
// 3 使用参数cmd 获取/设置锁 的操作都不是原子操作。

三、