终止网络连接的通常方法是调用close函数,不过close有两个限制,却可以使用shutdown来避免。
(1)close把描述符的引用计数减1,仅在该计数变为0时才关闭套接字,使用shutdown可以不管引用计数就激发TCP的正常连接终止序列。
(2)close终止读和写两个方向的数据传送。既然TCP连接是全双工的,有时我们需要告知对端我们已经完成了数据发送,即使对端任然有数据要发送给我们。下图展示了shutdown的常见调用场景:
#include <sys/socket.h>
int shutdown(int sockfd, int howto);
该函数的行为依赖于howto参数的值。
SHUT_RD 关闭连接着一半--套接字中不再有数据可接收,而且套接字接收缓冲区中的现有数据都被丢弃。进程不能再对这样的套接字调用任何读函数。对一个TCP套接字这样调用shutdown函数后,由该套接字接收的来自对端的任何数据都被确认,然后悄然丢弃。
SHUT_WR 关闭连接的写一半--对于TCP套接字,这称为半关闭。当前留在套接字发送缓冲区中的数据将被发送掉,后跟TCP的正常连接终止序列。我们已经说过,不管套接字描述符的引用计数量是否等于0,这样的写半部分关闭照样执行。进程不能再对这样的套接字调用任何写函数。
SHUT_EDWR 连接的读半部和写半部都关闭--这与调用shutdown两次等效:第一次调用指定SHUT_RD,第二次调用SHUT_WR。