iis服务器助手广告广告
返回顶部
首页 > 资讯 > 操作系统 >Linux网络I/O+Reactor模型是怎么样的
  • 225
分享到

Linux网络I/O+Reactor模型是怎么样的

2023-06-15 18:06:10 225人浏览 安东尼
摘要

本篇文章给大家分享的是有关linux网络I/O+Reactor模型是怎么样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。前言网络I/O,可以理解为网络上的数据流。通常我们会基

本篇文章给大家分享的是有关linux网络I/O+Reactor模型是怎么样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

前言

网络I/O,可以理解为网络上的数据流。通常我们会基于Socket与远端建立一条tcp或者UDP通道,然后进行读写。单个socket时,使用一个线程即可高效处理;然而如果是10K个socket连接,或者更多,我们如何做到高性能处理?

基本概念介绍

  • 进程(线程)切换

所有系统都有调度进程的能力,它可以挂起一个当前正在运行的进程,并恢复之前挂起的进程

  • 进程(线程)的阻塞

运行中的进程,有时会等待其他事件的执行完成,比如等待,请求I/O的读写;进程在等待过程会被系统自动执行阻塞,此时进程不占用CPU

  • 文件描述符

在Linux,文件描述符是一个用于表述指向文件引用的抽象化概念,它是一个非负整数。当程序打开一个现有文件或者创建一个新文件时,内核向进程返回一个文件描述符

  • linux信号处理

Linux进程运行中可以接受来自系统或者进程的信号值,然后根据信号值去运行相应捕捉函数;信号相当于是硬件中断的软件模拟

在零拷贝机制篇章已介绍过 用户空间和内核空间和缓冲区,这里就省略了

网络IO的读写过程

  • 当在用户空间发起对socket套接字的读操作时,会导致上下文切换,用户进程阻塞(R1)等待网络数据流到来,从网卡复制到内核;(R2)然后从内核缓冲区向用户进程缓冲区复制。此时进程切换恢复,处理拿到的数据

  • 这里我们给socket读操作的第一阶段起个别名R1,第二阶段称为R2

  • 当在用户空间发起对socket的send操作时,导致上下文切换,用户进程阻塞等待(1)数据从用户进程缓冲区复制到内核缓冲区。数据copy完成,此时进程切换恢复

linux五种网络IO模型

阻塞式I/O (blocking IO)

ssize_t recvfrom(int sockfd,void *buf,size_t len,unsigned int flags, struct sockaddr *from,socket_t *fromlen);

Linux网络I/O+Reactor模型是怎么样的

  • 最基础的I/O模型就是阻塞I/O模型,也是最简单的模型。所有的操作都是顺序执行的

  • 阻塞io模型中,用户空间的应用程序执行一个系统调用(recvfORM),会导致应用程序被阻塞,直到内核缓冲区的数据准备好,并且将数据从内核复制到用户进程。最后进程才被系统唤醒处理数据

  • 在R1、R2连续两个阶段,整个进程都被阻塞

非阻塞式I/O (nonblocking IO)

Linux网络I/O+Reactor模型是怎么样的

  • 非阻塞IO也是一种同步IO。它是基于轮询(polling)机制实现,在这种模型中,套接字是以非阻塞的形式打开的。就是说I/O操作不会立即完成,但是I/O操作会返回一个错误代码(EWOULDBLOCK),提示操作未完成

  • 轮询检查内核数据,如果数据未准备好,则返回EWOULDBLOCK。进程再继续发起recvfrom调用,当然你可以暂停去做其他事

  • 直到内核数据准备好,再拷贝数据到用户空间,然后进程拿到非错误码数据,接着进行数据处理。需要注意,拷贝数据整个过程,进程仍然是属于阻塞的状态

  • 进程在R2阶段阻塞,虽然在R1阶段没有被阻塞,但是需要不断轮询

多路复用I/O (IO multiplexing)

Linux网络I/O+Reactor模型是怎么样的

  • 一般后端服务都会存在大量的socket连接,如果一次能查询多个套接字的读写状态,若有任意一个准备好,那就去处理它,效率会高很多。这就是“I/O多路复用”,多路是指多个socket套接字,复用是指复用同一个进程

  • linux提供了select、poll、epoll等多路复用I/O的实现方式

  • select或poll、epoll是阻塞调用

  • 与阻塞IO不同,select不会等到socket数据全部到达再处理,而是有了一部分socket数据准备好就会恢复用户进程来处理。怎么知道有一部分数据在内核准备好了呢?答案:交给了系统系统处理吧

  • 进程在R1、R2阶段也是阻塞;不过在R1阶段有个技巧,在多进程、多线程编程的环境下,我们可以只分配一个进程(线程)去阻塞调用select,其他线程不就可以解放了吗

信号驱动式I/O (SIGIO)

Linux网络I/O+Reactor模型是怎么样的

  • 需要提供一个信号捕捉函数,并和socket套接字关联;发起sigaction调用之后进程就能解放去处理其他事

  • 当数据在内核准备好后,进程会收到一个SIGIO信号,继而中断去运行信号捕捉函数,调用recvfrom把数据从内核读取到用户空间,再处理数据

  • 可以看出用户进程是不会阻塞在R1阶段,但R2还是会阻塞等待

异步IO (POSIX的aio_系列函数)

Linux网络I/O+Reactor模型是怎么样的

  • 相对同步IO,异步IO在用户进程发起异步读(aio_read)系统调用之后,无论内核缓冲区数据是否准备好,都不会阻塞当前进程;在aio_read系统调用返回后进程就可以处理其他逻辑

  • socket数据在内核就绪时,系统直接把数据从内核复制到用户空间,然后再使用信号通知用户进程

  • R1、R2两阶段时进程都是非阻塞的

多路复用IO深入理解一波

select

int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);

1)使用copy_from_user从用户空间拷贝fd_set到内核空间

2)注册回调函数__pollwait

3)遍历所有fd,调用其对应的poll方法(对于socket,这个poll方法是sock_poll,sock_poll根据情况会调用到tcp_poll,udp_poll或者datagram_poll)

4)以tcp_poll为例,其核心实现就是__pollwait,也就是上面注册的回调函数

5)__pollwait的主要工作就是把current(当前进程)挂到设备的等待队列中,不同的设备有不同的等待队列,对于tcp_poll来说,其等待队列是sk->sk_sleep(注意把进程挂到等待队列中并不代表进程已经睡眠了)。在设备收到一条消息(网络设备)或填写完文件数据(磁盘设备)后,会唤醒设备等待队列上睡眠的进程,这时current便被唤醒了

6)poll方法返回时会返回一个描述读写操作是否就绪的mask掩码,根据这个mask掩码给fd_set赋值

7)如果遍历完所有的fd,还没有返回一个可读写的mask掩码,则会调用schedule_timeout是调用select的进程(也就是current)进入睡眠

8)  当设备驱动发生自身资源可读写后,会唤醒其等待队列上睡眠的进程。如果超过一定的超时时间(timeout指定),还是没人唤醒,则调用select的进程会重新被唤醒获得CPU,进而重新遍历fd,判断有没有就绪的fd

9)把fd_set从内核空间拷贝到用户空间

select的缺点

每次调用select,都需要把fd集合从用户态拷贝到内核态,这个开销在fd很多时会很大

同时每次调用select都需要在内核遍历传递进来的所有fd,这个开销在fd很多时也很大

select支持的文件描述符数量太小了,默认是1024

epoll

int epoll_create(int size);   int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);   int epoll_wait(int epfd, struct epoll_event *events,int maxevents, int timeout);
  • 调用epoll_create,会在内核cache里建个红黑树用于存储以后epoll_ctl传来的socket,同时也会再建立一个rdllist双向链表用于存储准备就绪的事件。当epoll_wait调用时,仅查看这个rdllist双向链表数据即可

  • epoll_ctl在向epoll对象中添加、修改、删除事件时,是在rbr红黑树中操作的,非常快

  • 添加到epoll中的事件会与设备(如网卡)建立回调关系,设备上相应事件的发生时会调用回调方法,把事件加进rdllist双向链表中;这个回调方法在内核中叫做ep_poll_callback

epoll的两种触发模式

epoll有EPOLLLT和EPOLLET两种触发模式,LT是默认的模式,ET是“高速”模式(只支持no-block socket)

  • LT(水平触发)模式下,只要这个文件描述符还有数据可读,每次epoll_wait都会触发它的读事件

  • ET(边缘触发)模式下,检测到有I/O事件时,通过 epoll_wait  调用会得到有事件通知的文件描述符,对于文件描述符,如可读,则必须将该文件描述符一直读到空(或者返回EWOULDBLOCK),否则下次的epoll_wait不会触发该事件

epoll相比select的优点

解决select三个缺点

  • 对于第一个缺点:epoll的解决方案在epoll_ctl函数中。每次注册新的事件到epoll句柄中时(在epoll_ctl中指定EPOLL_CTL_ADD),会把所有的fd拷贝进内核,而不是在epoll_wait的时候重复拷贝。epoll保证了每个fd在整个过程中只会拷贝一次(epoll_wait不需要复制)

  • 对于第二个缺点:epoll为每个fd指定一个回调函数,当设备就绪,唤醒等待队列上的等待者时,就会调用这个回调函数,而这个回调函数会把就绪的fd加入一个就绪链表。epoll_wait的工作实际上就是在这个就绪链表中查看有没有就绪的fd(不需要遍历)

  • 对于第三个缺点:epoll没有这个限制,它所支持的FD上限是最大可以打开文件的数目,这个数字一般远大于2048,举个例子,在1GB内存的机器上大约是10万左右,一般来说这个数目和系统内存关系很大

epoll的高性能

  • epoll使用了红黑树来保存需要监听的文件描述符事件,epoll_ctl增删改操作快速

  • epoll不需要遍历就能获取就绪fd,直接返回就绪链表即可

  • linux2.6 之后使用了mmap技术,数据不在需要从内核复制到用户空间,零拷贝

关于epoll的IO模型是同步异步的疑问

概念定义

  • 同步I/O操作:导致请求进程阻塞,直到I/O操作完成

  • 异步I/O操作:不导致请求进程阻塞,异步只用处理I/O操作完成后的通知,并不主动读写数据,由系统内核完成数据的读写

  • 阻塞,非阻塞:进程/线程要访问的数据是否就绪,进程/线程是否需要等待

异步IO的概念是要求无阻塞I/O调用。前面有介绍到I/O操作分两阶段:R1等待数据准备好。R2从内核到进程拷贝数据。虽然epoll在2.6内核之后采用mmap机制,使得其在R2阶段不需要复制,但是它在R1还是阻塞的。因此归类到同步IO

Reactor模型

Reactor的中心思想是将所有要处理的I/O事件注册到一个中心I/O多路复用器上,同时主线程/进程阻塞在多路复用器上;一旦有I/O事件到来或是准备就绪,多路复用器返回,并将事先注册的相应I/O事件分发到对应的处理器中

相关概念介绍:

  • 事件:就是状态;比如:读就绪事件指的是我们可以从内核读取数据的状态

  • 事件分离器:一般会把事件的等待发生交给epoll、select;而事件的到来是随机,异步的,所以需要循环调用epoll,在框架里对应封装起来的模块就是事件分离器(简单理解为对epoll封装)

  • 事件处理器:事件发生后需要进程或线程去处理,这个处理者就是事件处理器,一般和事件分离器是不同的线程

Reactor的一般流程

  • 1)应用程序在事件分离器注册读写就绪事件和读写就绪事件处理器

  • 2)事件分离器等待读写就绪事件发生

  • 3)读写就绪事件发生,激活事件分离器,分离器调用读写就绪事件处理器

  • 4)事件处理器先从内核把数据读取到用户空间,然后再处理数据

Linux网络I/O+Reactor模型是怎么样的

单线程 + Reactor

Linux网络I/O+Reactor模型是怎么样的

多线程 + Reactor

Linux网络I/O+Reactor模型是怎么样的

多线程 + 多个Reactor

Linux网络I/O+Reactor模型是怎么样的

Proactor模型的一般流程

1)应用程序在事件分离器注册读完成事件和读完成事件处理器,并向系统发出异步读请求

2)事件分离器等待读事件的完成

3)在分离器等待过程中,系统利用并行的内核线程执行实际的读操作,并将数据复制进程缓冲区,最后通知事件分离器读完成到来

4)事件分离器监听到读完成事件,激活读完成事件的处理器

5)读完成事件处理器直接处理用户进程缓冲区中的数据

Linux网络I/O+Reactor模型是怎么样的

Proactor和Reactor的区别

Proactor是基于异步I/O的概念,而Reactor一般则是基于多路复用I/O的概念

Proactor不需要把数据从内核复制到用户空间,这步由系统完成。

以上就是Linux网络I/O+Reactor模型是怎么样的,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注编程网操作系统频道。

--结束END--

本文标题: Linux网络I/O+Reactor模型是怎么样的

本文链接: https://www.lsjlt.com/news/281146.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Linux网络I/O+Reactor模型是怎么样的
    本篇文章给大家分享的是有关Linux网络I/O+Reactor模型是怎么样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。前言网络I/O,可以理解为网络上的数据流。通常我们会基...
    99+
    2023-06-15
  • 怎么解析Linux-I/O模型
    本篇文章为大家展示了怎么解析Linux-I/O模型,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。I/O介绍I/O通常有内存IO、网络I/O、磁盘I/O等,但我们通常说的是网络I/O以及磁盘I/O。网...
    99+
    2023-06-16
  • Java IO中Reactor网络模型的概念是什么
    小编给大家分享一下Java IO中Reactor网络模型的概念是什么,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!一、什么是 Reactor 模型:Th...
    99+
    2023-06-28
  • Redis单线程的reactor模型是怎样的
    这篇文章主要讲解了“Redis单线程的reactor模型是怎样的”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Redis单线程的reactor模型是怎样的”...
    99+
    2024-04-02
  • Java IO篇之Reactor 网络模型的概念
    目录一、什么是 Reactor 模型:二、Reactor 模型的分类:1、单 Reactor 单线程模型:1.1、处理流程:1.2、优缺点:2、单 Reactor 多线程模型:2.1...
    99+
    2024-04-02
  • Linux高性能网络IO和Reactor模型的示例分析
    小编给大家分享一下Linux高性能网络IO和Reactor模型的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!目录一、基本概念介绍二、网络IO的读写过程三...
    99+
    2023-06-20
  • MYSQL sync_relay_log对I/O thread的影响是怎样的
    MYSQL sync_relay_log对I/O thread的影响是怎样的,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。搭建好的一套从库...
    99+
    2024-04-02
  • Go语言是如何处理网络I/O操作的?
    欢迎各位小伙伴来到编程网,相聚于此都是缘哈哈哈!今天我给大家带来《Go语言是如何处理网络I/O操作的?》,这篇文章主要讲到等等知识,如果你对Golang相关的知识非常感兴趣或者正在自学,都可以关注我...
    99+
    2024-04-04
  • Linux的I/O机制怎么理解
    本篇内容主要讲解“Linux的I/O机制怎么理解”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Linux的I/O机制怎么理解”吧!你真的了解多线程吗如果问你“为什么多线程可以提高程序运行效率”,...
    99+
    2023-06-15
  • JAVA NIO怎么构建I/O多路复用的请求模型
    这篇文章主要介绍“JAVA NIO怎么构建I/O多路复用的请求模型”,在日常操作中,相信很多人在JAVA NIO怎么构建I/O多路复用的请求模型问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”JAVA NIO怎...
    99+
    2023-06-02
  • Linux的直接I/O机制怎么实现
    这篇文章主要介绍了Linux的直接I/O机制怎么实现的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Linux的直接I/O机制怎么实现文章都会有所收获,下面我们一起来看看吧。直接 I/O 的动机在介绍直接 I/O...
    99+
    2023-06-16
  • Linux下的网络IO模型怎么理解
    本篇内容介绍了“ Linux下的网络IO模型怎么理解”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!Redis,Nginx,Netty,Nod...
    99+
    2023-06-16
  • Kubernetes的网络模型是什么
    Kubernetes的网络模型是基于容器之间和容器与外部网络之间的通信。Kubernetes使用一个虚拟网络模型来为容器提供连接和通...
    99+
    2024-04-02
  • Linux网络编程中IO模型指的是什么
    今天就跟大家聊聊有关Linux网络编程中IO模型指的是什么,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。基于IO访问中存在的两个阶段详细介绍了Linux产生的五种IO模型。同步与异步...
    99+
    2023-06-06
  • C++ 中有哪些异步 I/O 模型?它们的优缺点是什么?
    C++ 中的异步 I/O 模型 异步 I/O 模型允许并发执行 I/O 操作,而不用阻塞主线程。这对于高性能应用程序至关重要,因为可以同时进行计算和其他活动。 C++ 中有两种主要类型...
    99+
    2024-05-08
    c++ 异步i/o
  • 怎么查看Linux系统的I/O调度器
    这篇文章主要介绍“怎么查看Linux系统的I/O调度器”,在日常操作中,相信很多人在怎么查看Linux系统的I/O调度器问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么查看Linux系统的I/O调度器”的疑...
    99+
    2023-06-16
  • 怎样简谈Kafka中的NIO网络通信模型
    这篇文章将为大家详细讲解有关怎样简谈Kafka中的NIO网络通信模型,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。摘要:很多人喜欢把RocketMQ与Kafka做对比,其实这两款消息队列的网...
    99+
    2023-06-02
  • Python怎样实现LeNet网络模型的训练及预测
    本篇文章给大家分享的是有关Python怎样实现LeNet网络模型的训练及预测,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。1.LeNet模型训练脚本整体的训练代码如下,下面我会...
    99+
    2023-06-21
  • Flink编程模型是怎样的
    这篇文章主要讲解了“Flink编程模型是怎样的”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Flink编程模型是怎样的”吧!抽象层次(Levels of Abstraction)Flink ...
    99+
    2023-06-02
  • Quick BI取数模型是怎么样的
    这篇文章主要介绍Quick BI取数模型是怎么样的,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!Quick BI 取数模型开发图表最关键的点在于选择准确的图表类型展示准确的数据,而准...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作