809-服务器编程(主线程与工作线程的分工)

服务器编程(主线程与工作线程的分工)

我们知道,服务器端为了能流畅处理多个客户端链接,一般在某个线程 A 里面 accept 新的客户端连接并生成新连接的 socket fd,然后将这些新连接的socketfd给另外开的数个工作线程 B1、B2、B3、B4,这些工作线程处理这些新连接上的网络 I/O 事件(即收发数据),同时,还处理系统中的另外一些事务。
这里我们将线程A称为主线程,B1、B2、B3、B4 等称为工作线程。工作线程的代码框架一般如下:

while (!m_bQuit)  
{  
    epoll_or_select_func();  
  
    handle_io_events();  
  
    handle_other_things();  
}  

在 epoll_or_select_func() 中通过 select() 或者 poll/epoll() 去检测 socket fd 上的 I/O 事件,若存在这些事件则下一步 handle_io_events() 来处理这些事件(收发数据),做完之后可能还要做一些系统其他的任务,即调用 handle_other_things()。

这样做有三个好处:
1、线程 A 只需要处理新连接的到来即可,不用处理网络IO事件。由于网络IO事件处理一般相对比较慢,如果在线程A里面既处理新连接又处理网络IO,则可能由于线程忙于处理IO事件,而无法及时处理客户端的新连接,这是很不好的。

2、 线程 A 接收的新连接,可以根据一定的负载均衡原则将新的 socket fd 分配给工作线程。常用的算法,比如 Round Robin(轮询机制),即,假设不考虑中途有连接断开的情况,一个新连接来了分配给 B1,又来一个分配给 B2,再来一个分配给 B3,再来一个分配给 B4。如此反复,也就是说线程A记录了各个工作线程上的 socket fd 数量,这样可以最大化地来平衡资源,避免一些工作线程“忙死”,另外一些工作线程“闲死”的现象。

3、即使工作线程不满载的情况下,也可以让工作线程做其他的事情。比如现在有4个工作线程,但只有3个连接。那么线程B4就可以在handle_other_thing()做一些其他事情。

效率问题

在上述 while 循环里面,epoll_or_selec_func() 中的 epoll_wait/poll/select 等函数一般设置了一个超时时间。如果设置超时时间为 0,那么在没有任何网络 I/O 事件和其他任务处理的情况下,这些工作线程实际上会空转,白白地浪费cpu时间片。如果设置的超时时间大于0,在没有网络 I/O 事件的情况,epoll_wait/poll/select 仍然要挂起指定时间才能返回,导致handle_other_thing()不能及时执行,影响其他任务不能及时处理,也就是说其他任务一旦产生,其处理起来具有一定的延时性。这样也不好。那如何解决该问题呢?

其实我们想达到的效果是,如果没有网络 I/O 事件和其他任务要处理,那么这些工作线程最好直接挂起而不是空转;如果有其他任务要处理,这些工作线程要立刻能处理这些任务而不是在 epoll_wait/poll/select 挂起指定时间后才开始处理这些任务。

我们采取如下方法来解决该问题,以 Linux 为例,不管 epoll_fd 上有没有文件描述符 fd,我们都给它绑定一个默认的 fd,这个 fd 被称为唤醒 fd。当我们需要处理其他任务的时候,向这个唤醒 fd 上随便写入 1 个字节的,这样这个 fd 立即就变成可读的了,epoll_wait()/poll()/select() 函数立即被唤醒,并返回,接下来马上就能执行 handle_other_thing(),其他任务得到处理。反之,没有其他任务也没有网络 I/O 事件时,epoll_or_select_func() 就挂在那里什么也不做。

这个唤醒 fd,在 Linux 平台上可以通过以下几种方法实现:
1.、管道 pipe,创建一个管道,将管道绑定到 epoll_fd 上。需要时,向管道一端写入一个字节,工作线程立即被唤醒。

2、 Linux 2.6 新增的 eventfd:

int eventfd(unsigned int initval, int flags); 

步骤也是一样,将生成的eventfd绑定到epoll_fd上。需要时,向这个eventfd上写入一个字节,工作线程立即被唤醒。陈硕大神写的C++ muduo网络库使用的就是eventfd。

3、第三种方法最方便。即 Linux 特有的 socketpair,socketpair 是一对相互连接的 socket,相当于服务器端和客户端的两个端点,每一端都可以读写数据。libevent网络库就是使用这种方法。

int socketpair(int domain, int type, int protocol, int sv[2]);

调用这个函数返回的两个 socket 句柄就是sv[0],和sv[1],在一个其中任何一个写入字节,在另外一个收取字节。
将收取的字节的 socket 绑定到 epoll_fd 上。需要时,向另外一个写入的 socket 上写入一个字节,工作线程立即被唤醒。

如果是使用 socketpair,那么 domain 参数一定要设置成 AFX_UNIX。

由于在 Windows 操作系统上,select 函数只支持检测 socket 这一种 fd,所以 Windows 上一般只能用方法 3 的原理。而且需要手动创建两个 socket,然后一个连接另外一个,将读取的那一段绑定到select的fd上去。这在写跨两个平台代码时,需要注意的地方。

文章借鉴来源:知乎:张小方

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

林林林ZEYU

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值