3.6.6.异步IO :SIGIO
3.6.6.1、何为异步IO
(1)几乎可以认为:异步IO就是操作系统用软件实现的一套中断响应系统。
(2)异步IO的工作方法是:我们当前进程注册一个异步IO事件(使用signal注册一个信号SIGIO的处理函数),然后当前进程可以正常处理自己的事情,当异步事件发生后当前进程会收到一个SIGIO信号从而执行绑定的处理函数去处理这个异步事件。
类似与中断
3.6.6.2、涉及的函数:
(1)fcntl(F_GETFL、F_SETFL、O_ASYNC、F_SETOWN)
F_GETFL 获取文件访问模式和文件状态标志*/
F_SETFL 设置 flag
O_ASYNC :可以接收异步通知
F_SETOWN : 设置异步信号来了,谁来通知
(2)signal或者sigaction(SIGIO)
3.6.3.代码实践
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <signal.h> /* SIGIO 异步信号声明头文件 */
int mousefd = -1;
//绑定 SIGIO 信号,在函数内 处理异步通知事件
void func(int sig)
{
char buf[200] = {0}; /* 定义一个buf */
if(sig != SIGIO)
return;
//读取鼠标
read(mousefd,buf,5); /* 此时fd 就是 mouse1 ,从fd (mouse1)中 读取 5 个 字节,放到 buf 中 */
//read 这个函数默认是 阻塞的, 如果我们不按键盘的 话,程序一直阻塞在这里
printf("鼠标读出的内容是:[%s] \n", buf);
}
int main(void)
{
int flag = -1;/* 定义设备文件描述符 */
char buf[200] = {0}; /* 定义一个buf */
mousefd = open("/dev/input/mouse1", O_RDONLY); /* open 打开mouse1 文件描述符, O_RDONLY 只读的文件打开 */
if (mousefd < 0) /* open 返回的 fd 小于 0,说明打开错误 */
{
perror("open:");
return -1;
}
//把鼠标的文件描述符设置为 可以接收 异步IO通知
flag = fcntl(mousefd, F_GETFL); /*F_GETFL : 获取 mousefd 文件描述符 flag */
flag |= O_ASYNC; /* mousefd 文件描述符 flag 添加 O_ASYNC :可以接收异步通知*/
fcntl(mousefd,F_SETFL, flag); /* F_SETFL 重新设置mousefd(已添加 O_ASYNC :可以接收异步通知) */
/* 以上3不 把鼠标的文件描述符设置为 可以接收 异步IO通知 */
//把异步IO事件的接收进程设置为 当前进程
fcntl(mousefd, F_SETOWN, getpid());
//注册当前进程 的SIGIO 信号捕获函数 读取鼠标
signal(SIGIO, func);
while(1)
{
//读取键盘
memset(buf, 0, sizeof(buf));/* 清空 buf */
read(0,buf,5); /* 从文件描述符 0 (0就是标准输入) 中 读取 2 个 字节,放到 buf 中 */
//read 这个函数默认是 阻塞的, 如果我们不按键盘的 话,程序一直阻塞在这里
printf("键盘读出的内容是:[%s] \n", buf);
}
return 0;
}
运行结果:
3.6.7.存储映射IO
3.6.7.1、mmap函数 : 内存映射
3.6.7.2、LCD显示和IPC之共享内存
3.6.7.3、存储映射IO的特点
(1)共享而不是复制,减少内存操作
(2)处理大文件时效率高,小文件不划算