客户端服务器TCP链接异常状态分析

1.服务器进程终止

当客户端和服务器建立连接后,kill服务器进程,此时服务器(子)进程所有打开的描述都被关闭,这导致服务器(子)进程向客户端发送一个FIN,而客户端则响应一个ACK。这是原TCP连接终止的前半部分工作。
此时,客户端仍可以向服务器端发送数据(服务器端无法向客户端发送数据,客户接收到FIN仅表示服务器进程已关闭了连接的服务器端,从而不再发生任何数据)。
当服务器端TCP接收到客户端的数据时,既然服务器进程已经终止,于是此时,其响应给客户端一个RST。
如果客户不理会服务器端发送的RST,继续向服务器端发送数据,即客户端继续向一个已接收到RST的套接字(客户端的套接字)执行写操作时,内核会向该进程发送一个SIGPIPE信号。该信号默认是终止进程,因此进程必须捕获它,以免它不情愿地被终止。
无论该进程是捕获了该信号并从处理函数返回还是简单的忽略该信号,写操作都将返回PIPE错误。

写操作数据是从进程空间到内核,读操作是内核到进程空间

2.服务器主机崩溃

当在不同的主机上运行客户端和服务器,然后断开服务器上的网络(服务器主机崩溃),此时客户端会持续重传数据分节,试图从服务器上接收一个ACK(源自Berkeley的实现重传该数据分节12次,共等待约9分钟才放弃重传),当客户端最终放弃的时候,服务器仍未重新启动或网络不可达,此时会给客户进程返回一个错误,假设客户进程阻塞在readline的调用上,那么该调用将返回错误。假设服务器主机崩溃,从而对客户分节根本没有响应,那么所返回的错误是ETIMEOUT。然而如果某个中间路由器判定服务器主机不可达,从而响应一个destination unreachable(目的主机不可达)ICMP消息,那么所返回的错误是EHOSTUNREACH或ENETUNREACH。
尽管我们的客户端最终还是会发现对端主机已崩溃或已不可达,不过有时候我们需要更快的检测出这种情况,这时所用的方法是对readline调用设置一个超时。
上面是通过客户端向服务器发送数据来检测出服务器端已经崩溃,若不想通过这种方式,可以使用SO_KEEPALIVE套接字选项获心博函数。

3.服务器主机崩溃后重启

服务器网络重新连通时,在这种情况下,当服务器主机崩溃后重启时,它的TCP连接丢失了崩溃前所有的连接信息,因此服务器TCP对所收到的来自客户的数据分节响应以一个RST。
当客户端收到RST时,客户正阻塞于readline的调用,导致该调用返回ECONNREST错误。


4、服务器主机关机

Unix系统关机时,init进程通常会先给所有进程发送SIGTERM信号(该信号可被捕获),等待一段固定时间后(往往在5到20s之间),然后给所有仍在运行的进程发送SIGKILL信号(该信号不能被捕获)。这么做留给所有运行的进程一小段时间来清除和终止。如果我们不能捕获SIGTERM信号并终止,我们的服务器将由SIGKILL信号终止。当服务器子进程终止时,它所有打开的描述符都将关闭,后续和服务器进程终止的情况一样。
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值