UNP学习总结(二)
本文是UNP复习系列的第二篇,主要包括了以下几个内容
- UNIX系统下5种I/O模型
- 阻塞、非阻塞,同步、异步
- epoll函数用例
一、Unix下的五种可用I/O模型
阻塞式I/O模型
阻塞式I/O是最简单的I/O模型。也是系统默认的I/O模型。

图中采用了
recvfrom(),使用TCP时候的read()时也是类似的。read()或者recvfrom()被作用于阻塞的文件描述符时,直到数据报到达且被复制到应用进程的缓冲区种或者发生错误时才返回,最常见的错误时被信号中断。非阻塞式I/O模型
非阻塞式I/O是在实际项目中最常用的I/O模型。对于很多进程或者线程而言,采用阻塞的I/O,有可能将该进(线)程阻塞在一个I/O操作上,这样的做法是不明智的。

On error, -1 is returned, and errno is set appropriately.
EAGAIN The file descriptor fd refers to a file other than a socket and has been marked nonblocking (O_NONBLOCK), and the read would block. See open(2) for further details on the O_NONBLOCK flag.
EAGAIN or EWOULDBLOCK The file descriptor fd refers to a socket and has been marked nonblocking (O_NONBLOCK), and the read would block.
以上内容引自
read()的man page。使用非阻塞I/O时,若没有数据立即可读,那么调用会立即返回错误,并且errno被设为EAGAIN或EWOULDBLOCK。对一个非阻塞的描述符反复调用read()或recvfrom()被称为轮询(polling)。轮询往往和多路复用配合使用。I/O复用模型
主要是指采用
select(),poll(),epoll()(Linux),kqueue()(BSD)等复用函数对多个描述符进行监听的方式。当采用这种方式时,进程阻塞在这些函数上,当注册于这些函数中的描述符允许读或写时函数返回。
在实践中,
select()、poll()由于性能及易用性等问题,已较少采用,大多数多路复用的场景都被epoll()函数取代,而带来的问题则是可移植性下降,因为epoll()只存在于Linux平台上。信号驱动式I/O模型
使用信号,让内核在描述符就绪式发送SIGIO通知进程读取的方式。

这种模型的优势在于等待数据报到达的器件进程不会被阻塞,主循环可以继续执行。
异步I/O模型
告知内核启动某个操作,并让内核在整个操作(包括将数据藏内核复制到我们自己的缓冲区)完成后通知我们。

这种模型的主要区别是不再需要调用
read()或recvfrom()进行读取操作,只需向异步I/O函数传入缓冲区指针和大小,内核将在负责把数据读入缓冲区,而后通知进程。这种模型应用很少,各种系统对其支持也不充分。这里补充一点内容。在较老的服务端模型中,采用阻塞I/O,每个连接被一个或多个线程管理是常见的方式。连接数量的增加就意味着线程数量的增加。这种模型对于几百上千的连接是可以胜任的,但对于更大量的连接,线程数量的增加导致线程间切换的开销也开始加大,耗费的内存等资源也开始增多,这种模型开始无法承受负担。
多路复用加上非阻塞I/O的方式则允许一个线程处理多个不同的连接,对多个描述符进行监听,并且只在描述符准备就绪时才对其进行处理。这种模型大大减少了非活跃线程的数量,也即减少了资源的开销。而由于一个线程中同时处理了多个描述符,那么使用阻塞I/O则可能使线程阻塞在其中一个描述符上,使得其它描述符无法得到处理。因此需要使用非阻塞的I/O。
二、阻塞和非阻塞,同步和异步
要注意阻塞/非阻塞和同步/异步的概念的区别。
POSIX把这两个属于定义如下:
同步I/O操作(synchronous I/O operation)导致请求进程阻塞,直到I/O操作完成;
异步I/O操作(asynchronous I/O operation)不导致请求进程阻塞。

前四种模型的主要区别在于第一阶段,其第二阶段是一样的:在数据从内核复制到调用者的缓冲区期间,进程阻塞于
recvfrom()调用。相反,异步I/O模型在这两个阶段都要处理,从而不同于其它四种模型。
阻塞不等于同步,非阻塞不等于异步,如在第二种模型中,采用的是非阻塞的I/O,但仍属于同步模型。
同步异步区分的关键在于将数据从内核空间复制回应用进程空间的这一过程是否阻塞进程。不管是阻塞还是非阻塞的I/O,当有数据可供消费时,在调用recvfrom()或read()将数据从内核复制回应用进程空间的这个过程中,该进(线)程是阻塞的,是被挂起的,因此他们都是同步的。只不过非阻塞I/O在无数据时,会立即返回,而非阻塞I/O不会。而异步的I/O连将数据复制到进程空间的这个过程,都不阻塞进程,而直接由内核完成,之后再以别的形式通知进程。
讲清楚这点应该就较好理解UNP种所说的:
根据上述定义,我们的前4种模型都是同步I/O模型,因为其中真正的I/O操作(recvfrom)将阻塞进程。只有异步I/O模型与POSIX定义的异步I/O相匹配。
三、epoll()用法总结
UNP中给出了select()和poll()的用法,这里不再重复。epoll()从Linux 2.5.44开始引入,目的是为了替代select()和poll(),达到更高的性能,实现更高的并发。这里总结一下它的用法。
man pages:
典型用法:
epoll_create()或epoll_create1()创建一个epoll实例- 设置需要监听的epoll_event中的事件类型以及描述符,并将其用
epoll_ctl()添加到epoll实例中 - 申请一个epoll_event数组,作为存放返回的活跃事件的容器
- 调用
epoll_wait(),传入epoll_event数组,阻塞等待 - 当
epoll_wait()返回时,返回值为epoll实例中活跃的事件个数,并且这些事件被写入到传入的epoll_event数组的前几个中。 - 处理活跃事件
- 返回第4步
epoll_event的定义如下:
typedef union epoll_data {
void *ptr;
int fd;
uint32_t u32;
uint64_t u64;
} epoll_data_t;
struct epoll_event {
uint32_t events; /* Epoll events */
epoll_data_t data; /* User data variable */
};
电平触发和边缘触发
- 若用作电平触发(默认),那么
epoll就只是一个速度更快的poll,它们的语法基本兼容,epoll可使用于任何poll的使用场景。 - 若使用了EPOLLET标志,那么epoll被用作边缘触发。采用边缘触发方式的描述符应该使用非阻塞I/O,以避免监听多个描述符的进程阻塞在其中一个描述符的
read()或write()操作中。 - 边缘触发的建议使用方式是(I)采用非阻塞描述符(II)等待
read()或write()返回EAGAIN。
程序示例
#include <stdio.h>
#include <stdlib.h>
#include <sys/epoll.h>
#include <sys/socket.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <arpa/inet.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <errno.h>
#define MAX_EVENTS 10
#define SERV_PORT 9877
#define BUFFER_SIZE 1024
int get_listen_fd();
void set_non_blocking(int fd);
void do_use_fd(int fd);
int main()
{
struct epoll_event ev, events[MAX_EVENTS];
int listen_sock, conn_sock, nfds, epollfd;
int n, addrlen;
struct sockaddr_in addr;
epollfd = epoll_create1(0);
listen_sock = get_listen_fd();
ev.events = EPOLLIN;
ev.data.fd = listen_sock;
epoll_ctl(epollfd, EPOLL_CTL_ADD, listen_sock, &ev);
for(;;){
nfds = epoll_wait(epollfd, events, MAX_EVENTS, -1);
for(n = 0; n <nfds; ++n){
if(events[n].data.fd == listen_sock){
conn_sock = accept(listen_sock, (struct sockaddr *) &addr, &addrlen);
if(conn_sock == -1){
if(errno != EAGAIN && errno != EWOULDBLOCK)
perror("accept");
break;
}
set_non_blocking(conn_sock);
ev.events = EPOLLIN | EPOLLET;
ev.data.fd = conn_sock;
epoll_ctl(epollfd, EPOLL_CTL_ADD, conn_sock, &ev);
}
else{
do_use_fd(events[n].data.fd);
}
}
}
return 0;
}
int get_listen_fd(){
int listen_fd;
struct sockaddr_in serv_addr;
listen_fd = socket(AF_INET, SOCK_STREAM, 0);
memset(&serv_addr, 0, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
serv_addr.sin_port = htons(SERV_PORT);
bind(listen_fd, (struct sockaddr*) &serv_addr, sizeof(serv_addr));
set_non_blocking(listen_fd);
listen(listen_fd, 10);
return listen_fd;
}
void set_non_blocking(int fd){
int s;
s = fcntl(fd, F_GETFL, 0);
s |= O_NONBLOCK;
fcntl(fd, F_SETFL, s);
}
void do_use_fd(int fd){
int done = 0;
for(;;){
int count;
char buf[BUFFER_SIZE];
count = read(fd, buf, sizeof(buf));
if(count == 0){
done = 1;
break;
}
else if(count == -1){
if(errno != EAGAIN){
perror("read");
done = 1;
}
break;
}
else{
write(STDOUT_FILENO, buf, count);
}
}
if(done){
printf("Closed connection on descriptor %d\n", fd);
close(fd);
}
}
示例说明
- listen socket 在本例中被设为边缘触发,且是非阻塞的,那么在返回
epoll_wait()等待前必须将所有到达的新连接全部处理了,直到EAGAIN被触发; - 本例中的
do_use_fd()也是类似的,边缘触发、非阻塞,返回等待前必须保证所有到达数据被处理; - 一个描述符被关闭,将导致epoll自动将该描述符对应的事件移除。不过要注意
dup2()或fork()等对描述符形成的复制。只有所有引用某个文件的描述符都被关闭了,epoll才会将相应的事件移除。(详见epoll的man page,Q6)
总结
几种I/O模型、阻塞/非阻塞常常是在服务器编程选型时候需要考虑的总点,本篇是相关基础知识的一个总结。熟悉相关的API和用法也是在编程时候提高效率,避免踩坑的必要条件。
UNP学习总结(二)的更多相关文章
- crawler4j 学习(二)
crawler4j 学习(二) 实现控制器类以制定抓取的种子(seed).中间数据存储的文件夹.并发线程的数目: public class Controller { public static voi ...
- 从零开始学习jQuery (二) 万能的选择器
本系列文章导航 从零开始学习jQuery (二) 万能的选择器 一.摘要 本章讲解jQuery最重要的选择器部分的知识. 有了jQuery的选择器我们几乎可以获取页面上任意的一个或一组对象, 可以明显 ...
- Android Animation学习(二) ApiDemos解析:基本Animators使用
Android Animation学习(二) ApiDemos解析:基本Animatiors使用 Animator类提供了创建动画的基本结构,但是一般使用的是它的子类: ValueAnimator.O ...
- AspectJ基础学习之二搭建环境(转载)
AspectJ基础学习之二搭建环境(转载) 一.下载Aspectj以及AJDT 上一章已经列出了他的官方网站,自己上去download吧.AJDT是一个eclipse插件,开发aspectj必装,他可 ...
- WPF的Binding学习笔记(二)
原文: http://www.cnblogs.com/pasoraku/archive/2012/10/25/2738428.htmlWPF的Binding学习笔记(二) 上次学了点点Binding的 ...
- AJax 学习笔记二(onreadystatechange的作用)
AJax 学习笔记二(onreadystatechange的作用) 当发送一个请求后,客户端无法确定什么时候会完成这个请求,所以需要用事件机制来捕获请求的状态XMLHttpRequest对象提供了on ...
- MyBatis学习系列二——增删改查
目录 MyBatis学习系列一之环境搭建 MyBatis学习系列二——增删改查 MyBatis学习系列三——结合Spring 数据库的经典操作:增删改查. 在这一章我们主要说明一下简单的查询和增删改, ...
- MyBatis学习 之 二、SQL语句映射文件(2)增删改查、参数、缓存
目录(?)[-] 二SQL语句映射文件2增删改查参数缓存 select insert updatedelete sql parameters 基本类型参数 Java实体类型参数 Map参数 多参数的实 ...
- MyBatis学习 之 二、SQL语句映射文件(1)resultMap
目录(?)[-] 二SQL语句映射文件1resultMap resultMap idresult constructor association联合 使用select实现联合 使用resultMap实 ...
- UML学习(二)-----类图
UML学习(二)-----类图 http://www.cnblogs.com/silent2012/archive/2011/09/07/2169946.html http://www.cnblogs ...
随机推荐
- 63、使用Timer类来实现定时任务
定时任务 定时任务就是让计算机自动的每隔一段时间执行的代码.比如要实现这样的一个功能:让计算机每隔5秒钟,在控制台打印一个www.monkey1024.com可以使用java.util包下的Timer ...
- 【codeforces】【比赛题解】#948 CF Round #470 (Div.2)
[A]Protect Sheep 题意: 一个\(R*C\)的牧场中有一些羊和一些狼,如果狼在羊旁边就会把羊吃掉. 可以在空地上放狗,狼不能通过有狗的地方,狼的行走是四联通的. 问是否能够保护所有的羊 ...
- python的面对对象
创建类 使用 class 语句来创建一个新类,class 之后为类的名称并以冒号结尾: class ClassName: '类的帮助信息' #类文档字符串 class_suite #类体 类的帮助信息 ...
- 【技巧总结】Penetration Test Engineer[2]-Information gathering
2.信息收集 信息收集是属于前期交互阶段所需要了解的问题. 2.1.前期交互内容 签署授权文件:首要要和受测试方签订授权协议. 划定范围:指定了一个二级域名作为测试目标,那么其他二级域名在测试范围内. ...
- Shell脚本中字符串判空:使用-z 字符串长度为0时,为真,-n字符串长度不为0,为真。这两个都不靠谱【转】
最近发现使用 -z 和 -n 来判断字符串判空,或不空时,很不靠谱. 使用下面的方法最可靠: if [ "x${value}" == "x" ] ...
- ansible批量修改linux服务器密码的playbook
从网上找到批量修改Linux服务器root密码的playbook. 使用方法: 1.输入要修改的inventory组 2.按需要,在playbook中输入要修改的IP.新密码,如下: - hosts: ...
- React-Native 之 常用组件Image使用
前言 学习本系列内容需要具备一定 HTML 开发基础,没有基础的朋友可以先转至 HTML快速入门(一) 学习 本人接触 React Native 时间并不是特别长,所以对其中的内容和性质了解可能会有所 ...
- python网络编程--事件驱动模型
论事件驱动与异步IO 事件驱动模型:根据事件的触发去干什么事,就是根据一个事件做反应 原文:http://www.cnblogs.com/alex3714/articles/5248247.html常 ...
- (一)问候 Jsoup
第一节: Jsoup 简介 Jsoup简介 jsoup 是一款Java 的HTML解析器,可直接解析某个URL地址.HTML文本内容.它提供了一套非常省力的API,可通过DOM,CSS以及类似于jQu ...
- 【读书笔记】Android平台的漏洞挖掘和分析
最近比较关注移动端的安全,以后也打算向安卓平台的安全发展.这篇博文主要是记录一些研究Android安全的读书笔记. Fuzzing技术的核心是样本生成技术 测试Android平台的组件间通信功能使用的 ...