本文讲述关于TCP连接关闭的系统状态、系统调用、报文。
客户端一般是主动关闭方,直接在程序在调用close()函数发出关闭请求(发送Fin报文)。
服务器端一般是关闭被动方,不会主动调用close()函数。
TCP连接关闭阶段的状态
当TCP其中一方调用close()函数之后,TCP连接进入四次挥手阶段。
FIN_WAIT_1
连接中断主动方的状态。
调用close()函数,发出连接中断请求FIN报文后进入该状态,等待对方的响应。
FIN_WAIT_2
连接中断主动方的状态。
收到了对方的中断请求ACK报文后进入该状态,等待对方的FIN报文。
TIME_WAIT
连接中断主动方的状态。
收到了对方的FIN报文,进入该状态,并发回最后一个ACK报文。然后等待一段时间(2倍的MSL),确保对方已收到该报文。
CLOSE_WAIT
连接中断被动方的状态。
收到对方的FIN报文后进入该状态,随后发出其最后需要发送的数据。
LAST_ACK
连接中断被动方的状态。
发送完最后需要发送的数据后,最后发送FIN报文,进入该状态,等待对方最后一个ACK报文。
CLOSED
连接中断被动方的状态。
收到对方最后一条ACK后,进入该状态。
关闭主动方的time_wait状态
关闭主动方最后会进入time_wait状态,这一状态会持续2msl。这么做的目的,一是确保对方收到了最后的ack报文;另一方面确保旧连接的报文经过这一时间段后全部作废。
四次挥手可以是三次
这种情况是:关闭被动方最后没有要发的数据了。于是直接把ack报文和fin报文合为一个报文发出。这样关闭主动方收到后,直接从fin_wait1状态变为time_wait状态(跳过了fin_wait2状态)。
连接的半关闭
关闭主动方调用shutdown函数,可以控制将本端的TCP连接的读或写关闭。
int shutdown(int sockfd,int how);
具体关闭方式由参数2来控制。
SHUT_RD :只写不读,接收到的数据会丢弃,不发送fin报文。
SHUT_WR :只读不写,输出缓冲区的数据会丢弃,要发送fin报文。
SHUT_RDWR :相当于调用shutdown两次:首先是以SHUT_RD,然后以SHUT_WR。
shutdown和close的区别:
如果多个进程共享一个套接字,close函数调用一次,套接字的计数引用减去1,为0时才关闭套接字,否则不会关闭套接字。
而shutdown不管引用计数。
进程如何关闭TCP连接?
1、对于客户端来说,一般调用close()来主动关闭连接。即使进程的代码中不调用close()函数,在进程终止时内核会自动关闭所有打开的描述符。
2、对于服务器来说,往往是TCP连接的被动方,不会主动调用close,那么进程需要能及时的感知到套接字的关闭。一般的,当服务器端发现与客户端通信的描述符不可读(读到了fin报文中的文件结束符eof)时,就认为连接关闭,这时候应用层,也就是服务器程序,把最后要发的数据发完,然后主动调用close()关闭这个描述符。
TCP连接的异常关闭
a、b两个正常连接的对端进程。假如b进程没有调用close就异常终止,那么发送FIN包是内核OS代劳,b进程已经不存在,当内核再次收到该socket的消息时,会回应RST报文(因为拥有该socket的进程已经终止)。a进程对收到RST的socket调用write时,内核会给a进程发送SIGPIPE,该信号默认情况下会终止当前a进程。
注意不管是怎么样描述符是如何被关闭的,都会使内核向描述符的对端发送FIN报文。