说在前面
- 环境: WSL
- 参考: UNIX网络编程、linux manual page
基本说明
通常Unix close函数也用来关闭套接字(其他如文件描述符),并终止TCP连接。
-
定义
#include <unistd.h> int close(int sockfd);
参数说明:
- sockfd
套接字描述符。
返回值:
- 成功返回0;出错则返回-1并置errno。
- sockfd
-
详细说明
- close函数关闭一个TCP套接字的默认行为是将该套接字标记成已关闭,然后立即返回到调用进程,该套接字描述符不能再由调用进程使用。但是,TCP将尝试发送已排列等待发送到对端的任何数据(即已写入套接字发送缓冲区的数据,见TCP)。发送完毕后发生的是TCP连接终止序列。
图-TCP连接终止序列 - 可通过设置SO_LINGER套接字选项来改变上述默认行为。
- close函数关闭一个TCP套接字的默认行为是将该套接字标记成已关闭,然后立即返回到调用进程,该套接字描述符不能再由调用进程使用。但是,TCP将尝试发送已排列等待发送到对端的任何数据(即已写入套接字发送缓冲区的数据,见TCP)。发送完毕后发生的是TCP连接终止序列。
描述符引用计数
-
在关于fork的讨论中,我们知道在调用fork后,父进程与子进程共享已打开的描述符,这个过程实际上会将所有共享的描述符的引用计数增加。
每个文件或者套接字都有一个引用计数。引用计数在文件表项中维护,它是当前打开着的引用该文件或套接字的描述符的个数。
-
在调用fork后,父进程通常会关闭已连接套接字(connfd),但是该操作只是将其描述符引用计数-1,此时计数大于0,那么父进程的此次close调用并不会引发TCP的连接终止序列。
-
如果需要在某个TCP连接中发送FIN(即引发TCP连接终止序列),可以使用shutdown函数。
-
若父进程不关闭已连接套接字描述符(connfd)
- 父进程将会耗尽可用描述符,因为任何进程在任何时刻拥有的打开的描述符是有限制的。
- 没有任何一个客户连接会被终止(就算在子进程关闭已连接套接字,父进程还是开着的,这样引用计数始终为1)