一、shutdown与close的区别
- 终止网络连接的通常方法是调用close函数。不过close有两个限制,却可以使用shutdown 来避免
close的两个限制
- ①close把描述符的引用计数减1,仅在该计数变为0时才关闭套接字。我们并发服务器模型文章中讨论过这一点。使用shutdown可以不管引用计数就激发TCP的正常连接终止序列
- ②close终止读和写两个方向的数据传送。既然TCP连接是全双工的,有时候我们需要告 知对端我们已经完成了数据发送,即使对端仍有数据要发送给我们。这就是我们在前一节中遇 到的str_cli函数在批量输入时的情况。下图展示了这样的情况下典型的函数调用
二、shutdown应用场景
- ①首先,只有最后一个活动引用关闭时,close才释放网络断点。这意味着如果复制一个奥杰自(如采用dup),要直到关闭了最后一个引用它的文件描述符才会释放这个套接字。而shutdown允许使一个套接字处于不活动状态,和引用它的文件描述符数目无关
- ②其次,有时可以很方便地关闭套接字双向传输的一个方向。例如,如果想让所有通信的进程能够确定数据传输何时结束,可以关闭该套接字的写端,然而通过该套接字读端仍可以继续接收数据
三、函数原型
#include <sys/socket.h>
int shutdown(int sockfd, int how);
//返回值:成功返回0;出错返回-1
参数1:
- 要关闭的描述符
参数2:
- SHUT_RD(数值0):关闭连接的读这一半——套接字中不再有数据可接收,而且套接字接收缓冲区 中的现有数据都被丢弃。进程不能再对这样的套接字调用任何读函数。对一个 TCP套接字这样调用shutdown函数后,由该套接字接收的来自对端的任何数据 都被确认,然后悄然丢弃
- 默认情形下,写入一个路由套接字中的所有数据都被作为同一个主 机上所有路由套接字的可能输入环回。有些程序把第二个参数指定为SHUT_RD来调 用shutdown函数以防止环回复制。防止环回复制的另一种方法是关闭 SO_USELOOPBACK套接字选项。
- SHUT_WR(数值1):关闭连接的写这一半——对于TCP套接字,这称为半关闭(half-close,见TCPv1 的18.5节)。当前留在套接字发送缓冲区中的数据将被发送掉,后跟TCP的正常 连接终止序列。我们已经说过,不管套接字描述符的引用计数是否等于0,这样 的写半部关闭照样执行。进程不能再对这样的套接字调用任何写函数
- SHUT_RDWR(数值2):连接的读半部和写半部都关闭——这与调用shutdown两次等效:第一次调用 指定SHUT_RD,第二次调用指定SHUT_WR