socket编程-优雅的断开链接shutdown

调用 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

  • SHUT_RD:断开输入流。套接字没法接收数据(即便输入缓冲区收到数据也被抹去),没法调用输入相关函数。
  • SHUT_WR:断开输出流。套接字没法发送数据,但若是输出缓冲区中还有未传输的数据,则将传递到目标主机。
  • SHUT_RDWR:同时断开 I/O 流。至关于分两次调用 shutdown(),其中一次以 SHUT_RD 为参数,另外一次以 SHUT_WR 为参数。

howto 在 Windows 下有如下取值:函数

  • SD_RECEIVE:关闭接收操做,也就是断开输入流。
  • SD_SEND:关闭发送操做,也就是断开输出流。
  • SD_BOTH:同时关闭接收和发送操做。

至于何时须要调用 shutdown() 函数,下节咱们会以文件传输为例进行讲解。code

close()/closesocket()和shutdown()的区别

确切地说,close() / closesocket() 用来关闭套接字,将套接字描述符(或句柄)从内存清除,以后不再能使用该套接字,与C语言中的 fclose() 相似。应用程序关闭套接字后,与该套接字相关的链接和缓存也失去了意义,TCP协议会自动触发关闭链接的操做。shutdown() 用来关闭链接,而不是套接字,无论调用多少次 shutdown(),套接字依然存在,直到调用 close() / closesocket() 将套接字从内存清除。
调用 close()/closesocket() 关闭套接字时,或调用 shutdown() 关闭输出流时,都会向对方发送 FIN 包。FIN 包表示数据传输完毕,计算机收到 FIN 包就知道不会再有数据传送过来了。
默认状况下,close()/closesocket() 会当即向网络中发送FIN包,无论输出缓冲区中是否还有数据,而shutdown() 会等输出缓冲区中的数据传输完毕再发送FIN包。也就意味着,调用 close()/closesocket() 将丢失输出缓冲区中的数据,而调用 shutdown() 不会。blog

相关文章
相关标签/搜索