Linux非阻塞IO(二)网络编程中非阻塞IO与IO复用模型结合
上文描述了最简易的非阻塞IO,采用的是轮询的方式,这节我们使用IO复用模型。
阻塞IO
过去我们使用IO复用与阻塞IO结合的时候,IO复用模型起到的作用是并发监听多个fd。
以简单的回射服务器为例,我们只监听了某fd是否可读,一旦fd有数据,我们立刻read,然后将其write给对方。
在阻塞IO里面,我们总是认为fd是可写的。因为即使底层的IO缓冲区已满,稍微等待片刻即可。这与read卡在一个无数据的fd上是两种情况。所以从这个角度出发,是不需要监听fd的写事件的。
总之,在阻塞IO中,收到数据然后write,这二者是同时使用的。
非阻塞IO
到了非阻塞IO里面,事情就远远不是这么简单了。此时,IO绝不仅仅是并发监听fd。
因为在这种情况下,如果某fd的write缓冲区满了,write会立刻返回-1,并且返回EWOULDBLOCK。所以数据的收和发未必可以同时进行,所以对于write操作,我们需要一个buffer来暂存数据。当fd可写时,才可以写入数据。
对于read一端,同样需要一个buffer,原因是因为,在阻塞IO中,假设双方协定好处理分包问题,对方发过来一个len为4000,然后我们需要调用readn函数确保收到足够的4000字节,这其中由于网络的拥塞,readn内部可能需要调用多次read系统调用。所以这中间需要短暂的等待。
到了非阻塞IO中,无法再使用readn反复调用read,否则就变成了轮询操作。如果数据收不满咋办?需要暂存起来,放到一个buffer中,由用户手工处理信息。
综上,非阻塞IO的read和write端都需要buffer。
详细的叙述请参考muduo库作者陈硕的Muduo 设计与实现之一:Buffer 类的设计
Buffer的设计
Buffer肯定要有输入和输出,所以我设计buffer的格式如下:
readindex表示要从Buffer中读取数据的起始位置。writeIndex则表示向Buffer中存放数据的起点。所以writeIndex – readIndex 是Buffer中数据的大小,而end – writeIndex 则表示Buffer中剩余的空间。
注意,每当buffer中的数据读完时,我们便重置指针,readIndex = writeIndex = begin。
使用非阻塞IO编写回射服务器的客户端
这里的逻辑是:用户从stdin输入数据,然后发给sockfd,随后从sockfd接收数据,输出给stdout。
所以这里我们要监听四次,stdin的读事件,sockfd的读和写事件,stdout的写事件。
注意这里需要两个buffer,因为存在两个数据流:
stdin -> Buffer –> sockfd
sockfd –> Buffer –> stdout
所以我们用到两个缓冲区:
一个用于向sockfd发送数据。
一个用于从sockfd接收数据。
后续将持续讲解Non-Blocking IO,直到完成完整的客户端和服务端。
未完待续。
Linux非阻塞IO(二)网络编程中非阻塞IO与IO复用模型结合的更多相关文章
- Java IO、网络编程、NIO、Netty、Hessian、RPC、RMI的学习路线
好久没看Java IO这块的内容,感觉都快忘得差不多了.平成编程也没有设计到太多的Java基础知识,所以这里希望可以抽点时间回顾一下,让艾宾浩斯记忆曲线不要下降的太快. 回顾这个主要还是以总结为主,能 ...
- linux下C语言socket网络编程简例
原创文章,转载请注明转载字样和出处,谢谢! 这里给出在linux下的简单socket网络编程的实例,使用tcp协议进行通信,服务端进行监听,在收到client的连接后,发送数据给client:clie ...
- 网络编程技术-----6、I/O复用实现并发服务器
网络编程技术-----6.I/O复用实现并发服务器 一.实验要求 服务器: 服务器等待接收客户的连接请求,一旦连接成功则显示客户地址,接着接收客户端的名称并显示:然后接收来自该客户的字符串,对 ...
- UNIX网络编程 第6章 I/O复用:select和poll函数
UNIX网络编程 第6章 I/O复用:select和poll函数
- Python网络编程(http协议,IO多路复用、select内核监听)
前言: 什么是IO? 分为IO设备和IO接口两个部分 如Linux系统,I/O操作可以有多种方式 比如DIO(DirectI/O) AIO(AsynchronousI/O异步I/O) Memory-M ...
- Java网络编程 -- BIO 阻塞式网络编程
阻塞IO的含义 阻塞(blocking)IO :阻塞是指结果返回之前,线程会被挂起,函数只有在得到结果之后(或超时)才会返回 非阻塞(non-blocking)IO :非阻塞和阻塞的概念相对应,指在不 ...
- Linux Linux程序练习十一(网络编程大文件发送UDP版)
//网络编程发送端--大文件传输(UDP) #include <stdio.h> #include <stdlib.h> #include <string.h> # ...
- Linux Linux程序练习十(网络编程大文件发送)
//网络编程客户端--大文件传输 #include <stdio.h> #include <stdlib.h> #include <string.h> #inclu ...
- linux驱动开发学习二:创建一个阻塞型的字符设备
在Linux 驱动程序中,可以使用等待队列来实现阻塞进程的唤醒.等待队列的头部定义如下,是一个双向列表. struct list_head { struct list_head *next, *pre ...
随机推荐
- 一种机制,与js类似
我们知道,当两个条件进行逻辑与操作的时候,其中任何一个条件为假,则表达式的结果为假.所以,遇到(A 且 B)这种表达式,如果A为假的话,B是不是真假都无所谓了,当遇到一个假条件的时候,程序也就没有必要 ...
- 插件 原生js 省市区 三级联动 源码
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- IAT Hook
@author: dlive 0X01 IAT Hook的优缺点 优点:工作原理与实现都比较简单 缺点:如果想钩取的API不在目标进程的IAT中,那么就无法使用该技术进行钩取操作.即如果要钩取的API ...
- log4j2 扩展日志级别,支持将系统日志与业务处理日志拆分
项目中,有时候需要对系统中已处理的一些业务数据日志进行提取分析,通常log4j默认提供的日志级别可能不够用,这时候我们就需要对日志级别进行扩展,以满足我们的需求. 本文就简单介绍一下log4j2的日志 ...
- 【Android开发日记】之入门篇(一)——开发环境的搭建
写给自己的话:至此,大学的时光已经剩下一年的时光,下一年等毕业设计结束后就算是正式地踏入社会.自己学android也不过几个月的时间,为了更好管理文档,写点东西记录下自己曾经做过的点点滴滴是一个不错的 ...
- hdu 2112(字典树+最短路)
HDU Today Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- NetStream论文
https://max.book118.com/html/2016/0102/32573670.shtm http://www.docin.com/p-1568348795.html
- 恢复安装过树莓派相关操作系统的TF卡容量
原文地址:传送门 前言玩树莓派的都知道,当我们向TF卡写入系统后,在Windows下能识别的只有几百M的容量了,这主要是由于在装Linux系统的时候给TF卡分了Windows无法识别的分区,下面我用图 ...
- [scrapy] 爬了第一页就停止 filtered off site request to
我碰到这个问题的原因是 allowed_domains 写错了 至于还有没有别的原因就不知道了
- 安装mysql数据库图文教程
一.首先下载该版本的Mysql 5.5.28双击软件,弹出软件的安装界面如下 二.点击Next ,点击同意