50.Reactor反应堆模式

使用ET+非阻塞,设计出一个正确IO的服务器

读取的时候要保证读取的数据来自对应的fd,发过来的报文即使读完了(ET模式必须读完)也大概率不是完整的(TCP面向字节流),但这个报文又不能丢,必须存着 -> 因此给每个fd,定义收发缓冲区 -> 把每个fd封装一下 --- Connection

listensocket和普通socket,有区别吗?本质上没有区别,只不过listensocket等待的是连接就绪,也是读就绪 -> listensocket使用继承的方式来实现统一

具体流程简图:

易错点1:

在send时,有两种情况:

1)_outbuffer写完了

2)_outbuffer没发完,内核发送缓冲区被占满了

**第二种情况:需要开启对于写事件的关心。**下次wait成功,再写如果还是满就继续开启写事件关心;再写如果写完了,就把对应的写事件关心去掉。

注意:EnableWREvent时,EPOLLET别写漏了;除了内核修改,Reactor内部的_connections也要修改。

易错点2:

实现回调机制时,每个Connection都要有一份。首先上层的listener注册回调,listener进行recv(accept)时,给每个新的Connection(Channel)都拷贝一份,这样每个Channel都有一份回调。

细节点1:

每个Connection内部都有一个Reactor类型的回指指针,便于操作_connections 和 _epoll_ptr进行用户和内核的数据修改。

多路转接的发送问题

接收缓冲区默认不就绪,发送缓冲区默认就绪(写事件默认就绪)。
结论:

1)读时间关心需要常设,而写事件关心是按需设置!如果写事件常设了,epoll就会一直写事件就绪

**2)写事件应该如何按需设置?如何发送?**直接发送,把发送缓冲区写满了(写事件不就绪),再把对写事件关心设置到内核。

3)手动开启对EPOLLOUT的关心,默认就要触发一次写事件就绪(ET也会触发)

Reactor多进程多线程实现方案分析

如果是用一个Reactor给多线程或者多进程用,那么它们可能epoll_wait就绪时拿到同一个文件描述符,多执行流同时向文件里写或读,会有线程安全问题。

最佳实践:一个执行流一个Reactor(One thread one Loop)

多进程实现方案

注:accept线程安全

1)主进程创建Reactor,创建listensock并添加后,创建一批子进程,有对应的管道连接(进程池)

2)主进程用epoll_wait时,有连接到来了,listensock读事件就绪了,但只是向对应的管道内发送revents,唤醒特定进程(负载均衡)。

3)特定进程被唤醒,读到revents,转而去listensock(从主进程继承下来的)去获取新连接添加到Reactor中,根据revents处理对应就绪的事件进行处理

多线程实现方案

1)主线程有一个Reactor,内部有listensock;创建一批线程,通过eventfd进行唤醒(负载均衡),各自有各自的Reactor

2)主线程epoll_wait只关心listensock的读事件,将获取到的sockfd入到一个队列中(加锁),唤醒一个特定线程(负载均衡)

3)被唤醒线程从加锁队列中读sockfd,然后添加到自己的Reactor,从此往后只关心自己Reactor的sockfd(子线程独自的Reactor内部没有listensock)

eventfd

  • eventfd 是一个轻量级的事件通知机制,基于文件描述符。
  • 它可以与 I/O 多路复用机制(如 epoll )结合使用。
  • 内核维护一个 64 位的计数器, write 会增加计数器, read 会减少计数器。

Reactor单进程版代码:

Linux-remote: linux远程仓库https://gitee.com/its-quite-six/linux-remote/tree/master/25_10_2/Reactor

相关推荐
A小辣椒2 天前
TShark:Wireshark CLI 功能
linux
A小辣椒2 天前
TShark:基础知识
linux
AlfredZhao2 天前
OCI 明明分配了 200G 系统盘,为什么 df 只看到 30G?
linux·oci
AlfredZhao2 天前
vi 删除指定范围的行,不用再反复按 dd
linux·vi
用户9718356334663 天前
银河麒麟 KY10 申威(SW64) 安装 nginx-1.16.1-2.p01.ky10.sw_64.rpm 详细步骤
linux
猪脚踏浪3 天前
linux 拷贝文件或目录到指定的位置
linux
摇滚侠3 天前
Linux CentOS7 rpm 安装 MySQL 5.7
linux·运维·mysql
bush43 天前
嵌入式linux学习记录十四、术语
linux·嵌入式
载数而行5203 天前
Linux 11 动态监控指令top
linux
不会C语言的男孩3 天前
Linux 系统编程 · 第 8 章:进程基础
linux·c语言