Linux shutdown与close

shutdown和close都可以用来关闭TCP连接, 那么它们有区别与联系呢?
下面来研究下这个问题:

1. close

关闭连接和套接字. 成功返回0; 失败为-1, errno被设置

#include <unistd.h>
int close(int fd);
  1. close会把描述符(sockfd)引用计数-1, 当计数为0时, 才真正断开连接、关闭套接字

  2. 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区别

  1. close关闭2个方向连接, 并释放连接对应资源(套接字), 而shutdown不会释放所有资源;
  2. close有引用计数的概念, 只有计数为0时, 才会释放套接字资源; shutdown没有引用计数, 是直接释放how指定的资源;
  3. close后, 只有引用计数为0时, 才会发送FIN分节; 以SHUT_WR或SHUT_RDWR为参数调用shutdown时, 总会发送FIN分节;

参考

TCP 的四次挥手,连接关闭的2种方式(close和shutdown) | CSDN

posted @ 2021-06-06 00:04  明明1109  阅读(437)  评论(0编辑  收藏  举报