IT技术博客大学习 共学习 共进步
全部 移动开发 后端 数据库 AI 算法 安全 DevOps 前端 设计 开发者

close_wait状态的产生原因及解决

Vimer 2009-12-11 22:59:48 累计浏览 3,174 次
本机暂存

最近需要上线的逻辑server由于需要与大量的后台server交互,今天突然发现有大量的close_wait产生,于是仔细研究了一下:
首先我们知道,如果我们的服务器程序APACHE处于CLOSE_WAIT状态的话,说明套接字是被动关闭的!
因为如果是CLIENT端主动断掉当前连接的话,那么双方关闭这个TCP连接共需要四个packet:

      Client -->  FIN  -->  Server 
      Client <--  ACK  <--  Server 
 这时候Client端处于FIN_WAIT_2状态;而Server 程序处于CLOSE_WAIT状态。
      Client <--  FIN  <--  Server 
这时Server 发送FIN给Client,Server 就置为LAST_ACK状态。
       Client -->  ACK  -->  Server 
Client回应了ACK,那么Server 的套接字才会真正置为CLOSED状态。

Server 程序处于CLOSE_WAIT状态,而不是LAST_ACK状态,说明还没有发FIN给Client,那么可能是在关闭连接之前还有许多数据要发送或者其他事要做,导致没有发这个FIN packet。
通常来说,一个CLOSE_WAIT会维持至少2个小时的时间(这个时间外网服务器通常会做调整,要不然太危险了)。如果有个流氓特地写了个程序,给你造成一堆的CLOSE_WAIT,消耗
你的资源,那么通常是等不到释放那一刻,系统就已经解决崩溃了。
只能通过修改一下TCP/IP的参数,来缩短这个时间:修改tcp_keepalive_*系列参数有助于解决这个问题。

但是实际上,还是主要是因为我们的程序代码有问题,通常是如下问题:
比如被动关闭的是客户端。。。

当对方调用closesocket的时候,你的程序正在

int nRet = recv(s,….); 
if (nRet == SOCKET_ERROR) 

    // closesocket(s); 
    return FALSE; 
}

很多人就是忘记了那句closesocket,这种代码太常见了。

我的理解,当主动关闭的一方发送FIN到被动关闭这边后,被动关闭这边的TCP马上回应一个ACK过去,同时向上面应用程序提交一个ERROR,导 致上面的SOCKET的send或者recv返回SOCKET_ERROR,正常情况下,如果上面在返回SOCKET_ERROR后调用了 closesocket,那么被动关闭的者一方的TCP就会发送一个FIN过去,自己的状态就变迁到LAST_ACK.

同分类推荐文章

  1. 等了十年的 Go 链式管道,终于来了:seq 让你像写 Scala 一样写 Go (2026-06-25 18:38:18)
  2. Go 实验特性详解 (2026-06-21 10:05:27)
  3. amd64 微架构级别对 Go 程序性能提升多少? (2026-06-21 09:38:49)

查看更多 后端 文章 →

建议继续学习

  1. 如何成为Python高手 (累计阅读 54,992)
  2. gen_tcp发送进程被挂起起因分析及对策 (累计阅读 37,821)
  3. TCP 的那些事儿(上) (累计阅读 22,696)
  4. 从输入 URL 到页面加载完成的过程中都发生了什么事情? (累计阅读 15,932)
  5. libcurl的使用总结(二) (累计阅读 15,083)
  6. 自建DNS以防止GFW干扰 (累计阅读 13,125)
  7. Linux 性能监控、测试、优化工具 (累计阅读 13,010)
  8. include(“./file.php”)和include(“file.php”)区别 (累计阅读 12,788)
  9. Rolling cURL: PHP并发最佳实践 (累计阅读 11,486)
  10. 浅谈TCP优化 (累计阅读 11,079)