1 close函数
通常的Unix close函数也用来关闭套接字,并终止TCP连接。
#include <unistd.h>
int close(int sockfd);
close将描述符引用计数减1。当该计数减到0时,对TCP套接字而言默认行为是把该套接字标记成已关闭,然后立即返回到调用进程。已关闭套接字不能再由调用进程使用,也就是说它不能再作为read或write的第一个参数。然而TCP将尝试发送已排队等待发送到对端的任何数据,发送完毕后发生的是正常的TCP连接终止序列。
SO_LINGER套接字选项可以用来改变TCP套接字的这种默认行为。
2 shutdown函数
终止网络连接的通常方法是调用close函数。不过close有两个限制,却可以使用shutdown来避免。
(1)close把描述符的引用计数减1,仅在该计数变为0时才关闭套接字。使用shutdown可以不管引用计数就激发TCP的正常连接终止序列。
(2)close终止读和写两个方向的数据传送。既然TCP连接是全双工的,有时候我们需要告知对端我们已经完成了数据发送,即使对端仍有数据要发送给我们。图1展示了这样的情况下典型的函数调用。
图1 调用shutdown关闭一半TCP连接
#include <sys/socket.h>
int shutdown(int sockfd, int howto);
返回值:若成功则返回0,否则返回-1
该函数的行为依赖于howto参数的值:
SHUT_RD 关闭连接的读这一半--套接字中不再有数据可接收,而且套接字接收缓冲区中的现有数据都被丢弃。进程不能再对这样的套接字调用任何读函数。对一个TCP套接字这样调用shutdown函数后,由该套接字接收的来自对端的任何数据都被确认,然后悄然丢弃。
SHUT_WR 关闭连接的写这一半--对于TCP套接字,这称为半关闭(half-close)。当前留在套接字发送缓冲区中的数据将被发送掉,后跟TCP的正常连接终止序列。我们已经说过,不管套接字描述符的引用计数是否等于0,这样的写半部关闭照样执行。进程不能再对这样的套接字调用任何写函数。
SHUT_RDWR 连接的读半部和写半部都关闭--这与调用shutdown两次等效:第一次调用指定SHUT_RD,第二次调用指定SHUT_WR。