转的,没验证

close(sock_fd)会把sock_fd的内部计数器减1
当sock_fd的内部计数器为0时, 才调用shutodwn(), 并最终释放文件描述符
调用shutdown()只是进行了TCP断开, 并没有释放文件描述符

本来正常的TCP程序不需要显示调用shutdown()
但某些TCP程序十分不友好, 包括著名的firefox早期版本, 给服务器吃CLOSE_WAIT
调用shutodwn()就不会CLOSE_WAIT, 只会FIN_WAIT1或FIN_WAIT2

这就是服务器没有调用shutdown引起的
客户端非正常退出会给服务器带来CLOSE_WAIT

CLOSE_WAIT非常讨厌, 会卡住(阻塞)close()函数

以下描述主要是针对windows平台下的TCP socket而言。

首先需要区分一下关闭socket和关闭TCP连接的区别,关闭TCP连接是指TCP协议层的东西,就是两个TCP端之间交换了一些协议包(FIN,RST等),具体的交换过程可以看TCP协议,这里不详细描述了。而关闭socket是指关闭用户应用程序中的socket句柄,释放相关资源。但是当用户关闭socket句柄时会隐含的触发TCP连接的关闭过程。

TCP连接的关闭过程有两种,一种是优雅关闭(graceful close),一种是强制关闭(hard close或abortive close)。所谓优雅关闭是指,如果发送缓存中还有数据未发出则其发出去,并且收到所有数据的ACK之后,发送FIN包,开始关闭过程。而强制关闭是指如果缓存中还有数据,则这些数据都将被丢弃,然后发送RST包,直接重置TCP连接。

下面说一下shutdown及closesocket函数。

shutdown函数的原型是:

int shutdown(

SOCKET s,

int how

);

该函数用于关闭TCP连接,单并不关闭socket句柄。其第二个参数可以取三个值:SD_RECEIVE,SD_SEND,SD_BOTH。

SD_RECEIVE表明关闭接收通道,在该socket上不能再接收数据,如果当前接收缓存中仍有未取出数据或者以后再有数据到达,则TCP会向发送端发送RST包,将连接重置。

SD_SEND表明关闭发送通道,TCP会将发送缓存中的数据都发送完毕并收到所有数据的ACK后向对端发送FIN包,表明本端没有更多数据发送。这个是一个优雅关闭过程。

SD_BOTH则表示同时关闭接收通道和发送通道。

From: http://blog.csdn.net/bad_sheep/article/details/6157738

closesocket函数的原型是:

int closesocket(

SOCKET s

);

该函数用于关闭socket句柄,并释放相关资源。前面说过,关闭socket句柄时会隐含触发TCP连接的关闭过程,那么closesocket触发的是一个优雅关闭过程还是强制关闭过程呢?这个与一个socket选项有关:SO_LINGER 选项,该选项的设置值决定了closesocket的行为。该选项的参数值是linger结构,其定义是:

typedef struct linger {

u_short l_onoff;

u_short l_linger;

} linger;

当l_onoff值设置为0时,closesocket会立即返回,并关闭用户socket句柄。如果此时缓冲区中有未发送数据,则系统会在后台将这些数据发送完毕后关闭TCP连接,是一个优雅关闭过程,但是这里有一个副作用就是socket的底层资源会被保留直到TCP连接关闭,这个时间用户应用程序是无法控制的。

当l_onoff值设置为非0值,而l_linger也设置为0,那么closesocket也会立即返回并关闭用户socket句柄,但是如果此时缓冲区中有未发送数据,TCP会发送RST包重置连接,所有未发数据都将丢失,这是一个强制关闭过程。

当l_onoff值设置为非0值,而l_linger也设置为非0值时,同时如果socket是阻塞式的,此时如果缓冲区中有未发送数据,如果TCP在l_linger表明的时间内将所有数据发出,则发完后关闭TCP连接,这时是优雅关闭过程;如果如果TCP在l_linger表明的时间内没有将所有数据发出,则会丢弃所有未发数据然后TCP发送RST包重置连接,此时就是一个强制关闭过程了。

另外还有一个socket选项SO_DONTLINGER,它的参数值是一个bool类型的,如果设置为true,则等价于SO_LINGER中将l_onoff设置为0。

注意SO_LINGER和SO_DONTLINGER选项只影响closesocket的行为,而与shutdown函数无关,shutdown总是会立即返回的。

所以为了保证建议的最好的关闭方式是这样的:

发送完了所有数据后:

(1)调用shutdown(s, SD_SEND),如果本端同时也接收数据时则执行第二步,否则跳到第4步。

(2)继续接收数据,

(3)收到FD_CLOSE事件后,调用recv函数直到recv返回0或-1(保证收到所有数据),

(4)调用closesocket,关闭socket句柄。

在实际编程中,我们经常也不调用shutdown,而是直接调用closesocket,利用closesocket隐含触发TCP连接关闭过程的特性。此时的过程就是:

当发送完所有数据后:

(1)如果本端同时也接受数据则则执行第二步,否则跳到第4步。

(2)继续接收数据,

(3)收到FD_CLOSE事件后,调用recv函数直到recv返回0或-1(保证收到所有数据),

(4)调用closesocket,关闭socket句柄。

但是此时为了保证数据不丢失,则需要设置SO_DONTLINGER选项,不过windows平台下这个也是默认设置。

经过实验发现,发送端应用程序即便是异常退出或被kill掉进程,操作系统也不会丢弃发送缓冲区中的未发送数据,而是会在后台将这些数据发送出去。但是这是在socket的发送缓存不为0的前提下,当socket的发送缓存设置为0(通过SO_SNDBUF选项)时比较特殊,此时不论socket是否是阻塞的,send函数都会被阻塞直到传入的用户缓存中的数据都被发送出去并被确认,因为此时在驱动层没有分配缓存存放用户数据,而是直接使用的应用层的用户缓存,所以必须阻塞直到数据都发出,否则可能会造成系统崩溃。

另外,如果是接收端的应用程序异常退出或被kill掉进程,并且接收缓存中还有数据没有取出的话,那么接收端的TCP会向发送端发送RST包,重置连接,因为后续数据已经无法被提交应用层了。

最后这里说一个感觉是windows的bug,就是做这样的一个测试:

在一端线listen一个socket,然后在另一端connect,connect成功后,listen端会检测到网络事件触发,在listen端accept之前,将connect端kill掉,然后继续运行listen端,listen端任然会accept成功,且在accept出来的socket发送数据也能成功。发送完之后在等网络事件,此时又会等待成功,但是调用WSAEnumNetworkEvents得出的事件标识却是0。之后再也不会等到网络事件。

转载于:https://www.cnblogs.com/liyulong1982/p/3990740.html

close和shutdown的区别相关推荐

  1. socket关闭close和shutdown的区别

    socket关闭close和shutdown的区别 close--只是减少该socket id的引用计数,并不直接关闭网络连接,其他使用这个socket id的进程还可以正常进行网络通信的读写操作.直 ...

  2. halt ,poweroff和shutdown 的区别

    在linux下一些常用的关机/重启命令有shutdown.halt.reboot.及init,它们都可以达到重启系统的目的,但每个命令的内部工作过程是不同的,通过本文的介绍,希望你可以更加灵活的运用各 ...

  3. 关闭线程池 shutdown 和 shutdownNow 的区别

    欢迎关注方志朋的博客,回复"666"获面试宝典 来源:https://blog.csdn.net/xiewenfeng520/article/details/107013342 前 ...

  4. 正确关闭线程池:shutdown 和 shutdownNow 的区别

    点击关注公众号,实用技术文章及时了解 来源:blog.csdn.net/xiewenfeng520/ article/details/107013342 前言 本章分为两个议题 如何正确关闭线程池 s ...

  5. 关闭线程池 shutdown 和 shutdownNow 的区别?

    前言 本章分为两个议题 如何正确关闭线程池 shutdown 和 shutdownNow 的区别 1.线程池示例 public class ShutDownThreadPoolDemo {privat ...

  6. CentOS关机大法之shutdown应用实例

    2019独角兽企业重金招聘Python工程师标准>>> 在执行命令shutdown -h 10 'System will shutdown in 10 minitus,please ...

  7. shutdown()函数:优雅地断开TCP连接

    调用 close()/closesocket() 函数意味着完全断开连接,即不能发送数据也不能接收数据,这种"生硬"的方式有时候会显得不太"优雅". 图1:cl ...

  8. 网络编程之shutdown() 与 close()函数详解

    1.close()函数 [cpp] view plaincopyprint? <span style="font-size:13px;">#include<uni ...

  9. java并发核心知识体系精讲_JAVA核心知识汇总

    双非同学如何逆袭大厂? 在互联网行业,入行的第一份工作很大程度上决定了以后职业发展的高度.有些双非的同学认为自己校招进不了大厂以后还会有社招,这种想法很危险.大厂的社招,大多数都只招大厂的员工.什么意 ...

最新文章

  1. 一个简单的python爬虫(转)
  2. bzoj1816: [Cqoi2010]扑克牌
  3. sublime text使用小技巧
  4. JAVA基础加强(张孝祥)_类加载器、分析代理类的作用与原理及AOP概念、分析JVM动态生成的类、实现类似Spring的可配置的AOP框架...
  5. 实现 scrollview 默认显示指定的页码
  6. For each...in / For...in / For...of 的解释与例子
  7. 7-69 超市促销 (6 分)
  8. maven入门(7)maven项目(组件)的坐标
  9. 学习笔记 再论静态方法和类方法
  10. 使用.NET和Jquery打造简单的便签纸
  11. mplayer命令行模式下的使用方法
  12. TypeError: Cannot read property ‘forceUpdate‘ of undefined
  13. 灰度到伪彩色的转换公式
  14. 免费的短信验证码接口
  15. 灵飞经5龙生九子 第二十一章 危机四伏 3
  16. learn the python in hard way习题16~20的附加习题
  17. frame-reday
  18. android 复制asset文件下文件到手机指定路径,手机根目录
  19. 易语言如何调用c dll文件,易语言调用C++写的DLL
  20. Android软键盘使用

热门文章

  1. linux下的科学软件下载,十分科学app-十分科学官网版下载v1.4.3-Linux公社
  2. JDK注解(内置和自定义)
  3. 003_SQL语言简介
  4. php简单分词,php版简易中文分词实现 – mysql版本
  5. Kotlin 系统入门到进阶
  6. HTTP代理神器Fidder
  7. Java基础:常用IO流
  8. Mysql数据库中修改库名的的方法
  9. python随机生成数字_Python 中生成 0 到 9 之间的随机整数
  10. plsql oracle服务端,PLSQL Developer + Oracle客户端简易安装方法