89-非阻塞 accept

不知道你是否还记得异常连接—— accept 返回前连接终止。这篇文章探讨的是在 accept 函数调用前,连接被异常终止的情况。

不过,很遗憾我们并未观察到 accept 产生异常。man 手册中解释,linux 会把错误转移到 accept 返回的新套接字(已连接套接字)中,因此在后续的 read 已连接套接字的时候,会产生错误。

假设我考虑最坏的情况,即 accept 会产生错误,会怎样呢?

1. 容易产生错误的代码

while(1) {
   rfds = {listenfd, ...}
   select(maxfd + 1, &rfds, ...);
   if (listenfd in rfds) {
      // 可能会阻塞
      accept(listenfd, ...);
   }
}

有些系统的实现中,如果在 select 返回后,accept 调用前,连接异常终止,内核会把该连接从已连接队列中删除,并释放相应的已连接套接字。如此一来,就好像连接从未发生过一样,这时候调用 accept 就可能会阻塞。

另外有些系统会让 accept 直接返回一个 ECONNABORTED 错误或 EPROTO 错误。

虽然在 Linux 中并不会出现这种情况……

2. 使用非阻塞 accept

为了防止 accept 被阻塞,应该使用非阻塞 i/o. 这样就可以捕捉到 EWOULDBLOCK 错误,让程序继续下去。

// 设置 listenfd 为非阻塞套接字。
setNonblock(listefd, 1);
while(1) {
   rfds = {listenfd, ...}
   select(maxfd + 1, &rfds, ...);
   if (listenfd in rfds) {
      ret = accept(listenfd, ...);
      if (ret < 0) {
        // 如果是 EWOULDBLOCK 则继续循环
        if (errno == EWOULDBLOCK) continue;
        ERR_EXIT("accept");
      }
   }
}

3. 总结

  • 知道 accept 返回前连接异常终止该如何处理
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值