本文讨论在网络套接字编程中,服务器终止的几种情况。
1.服务器进程终止
服务器和客户端建立连接之后,找到服务器子进程的ID,kill杀死之后,作为进程终止处理的部分工作,子进程中所有打开的描述符都被关闭。从而导致向客户发送一个FIN,而客户TCP则响应一个ACK。以上是TCP连接终止工作的前半部分。
当客户端受到FIN则表示服务器进程已经关闭了,从而服务器不会再发送任何数据。
1.1 当服务器进程终止后第一次收到客户端数据
当服务器进程终止后,第一次收到客户端的数据,服务器不知该怎么处理,则返回RST给客户端。
1.2 当服务器进程终止后第二次收到客户端数据
当服务器进程终止后,客户端向其第二次发送数据,则客户端会收到来自内核的SIGPIPE信号。
2. 服务器崩溃
服务器奔溃不同于操作指令终止,其是突然发生,而来不及发送相关的信息给客户端。
所以对于客户端来说,还不知道服务器的状态,以为其处于正常状态。
此时如果客户端向服务器发送数据,则会进行持续重传,试图获取服务器的ACK。
最终,客户端收到timeout的错误信息,或者存在中间路由器的话,收到ICMP不可达信息。
3.服务器主机奔溃后重启
当服务器主机奔溃后重启,其TCP丢失了奔溃前的所有连接信息,因此服务器TCP对于所收到的来自客户端的数据响应RST。
3.服务器主机正常关机
正常关机不同于被kill, UNIX正常关机时,init进程通常先给所有进程发送SIGTERM信号,等待一段固定的时间,然后给所有仍在运行的进程发送SIGKILL信号。
客户端使用select和poll函数,使得可以检测服务器进程终止的信息,从而不会再误向其发送数据。