调用 close()/closesocket() 函数意味着彻底断开链接,即不能发送数据也不能接收数据,这种“生硬”的方式有时候会显得不太“优雅”。
上图演示了两台正在进行双向通讯的主机。主机A发送完数据后,单方面调用 close()/closesocket() 断开链接,以后主机A、B都不能再接受对方传输的数据。实际上,是彻底没法调用与数据收发有关的函数。缓存
通常状况下这不会有问题,但有些特殊时刻,须要只断开一条数据传输通道,而保留另外一条。
使用 shutdown() 函数能够达到这个目的,它的原型为:网络
int shutdown(int sock, int howto); //Linux int shutdown(SOCKET s, int howto); //Windows
sock 为须要断开的套接字,howto 为断开方式。socket
howto 在 Linux 下有如下取值:ide
howto 在 Windows 下有如下取值:函数
至于何时须要调用 shutdown() 函数,下节咱们会以文件传输为例进行讲解。code
确切地说,close() / closesocket() 用来关闭套接字,将套接字描述符(或句柄)从内存清除,以后不再能使用该套接字,与C语言中的 fclose() 相似。应用程序关闭套接字后,与该套接字相关的链接和缓存也失去了意义,TCP协议会自动触发关闭链接的操做。shutdown() 用来关闭链接,而不是套接字,无论调用多少次 shutdown(),套接字依然存在,直到调用 close() / closesocket() 将套接字从内存清除。
调用 close()/closesocket() 关闭套接字时,或调用 shutdown() 关闭输出流时,都会向对方发送 FIN 包。FIN 包表示数据传输完毕,计算机收到 FIN 包就知道不会再有数据传送过来了。
默认状况下,close()/closesocket() 会当即向网络中发送FIN包,无论输出缓冲区中是否还有数据,而shutdown() 会等输出缓冲区中的数据传输完毕再发送FIN包。也就意味着,调用 close()/closesocket() 将丢失输出缓冲区中的数据,而调用 shutdown() 不会。blog