shutdown和close都可以用来关闭TCP连接, 那么它们有区别与联系呢?
下面来研究下这个问题:
1. close
关闭连接和套接字. 成功返回0; 失败为-1, errno被设置
#include <unistd.h>
int close(int fd);
-
close会把描述符(sockfd)引用计数-1, 当计数为0时, 才真正断开连接、关闭套接字
-
close终止读、写两个方向的数据传送
- 对读方向, 内核将套接字设置为不可读, 任何读操作都会返回异常;
- 对写方向, 内核尝试将发送缓冲区中的数据发送给对端, 最后发送FIN分节, 接下来如果再对该套接字进行写操作会返回异常;
- 如果对端还继续发送数据, 就会响应RST分节;
注意: close会关闭2个方向的连接
2. shutdown
关闭连接的一个方向, 具体取决于参数how. 成功返回0; 失败为-1, errno被设置
#include <sys/socket.h>
int shutdown(int sockfd, int how);
how有3个值:
-
SHUT_RD 关闭连接的读方向, 不能再通过套接字读取数据(读操作会直接返回EOF), 而且套接字接收缓冲区现有数据都被丢弃. 如果再收到新数据, 会对数据进行ACK, 然后直接丢弃.
-
SHUT_WR 关闭连接的写方向, 常称为"半关闭 half-close". 现有发送缓冲区的数据将会被发送掉, 后跟FIN分节. 如果程序对该套接字进行写操作, 会报错.
-
SHUT_RDWR 连接的读半部和写半部都关闭, 与调用shutdown两次等效: shutdown(sockfd, SHUT_RD) + shutdown(sockfd, SHUT_WR).
3. shutdown和close区别
- close关闭2个方向连接, 并释放连接对应资源(套接字), 而shutdown不会释放所有资源;
- close有引用计数的概念, 只有计数为0时, 才会释放套接字资源; shutdown没有引用计数, 是直接释放how指定的资源;
- close后, 只有引用计数为0时, 才会发送FIN分节; 以SHUT_WR或SHUT_RDWR为参数调用shutdown时, 总会发送FIN分节;