后端面试之网络编程-close和shutdown的区别
本篇博客的视频教程首发于 Youtube:科技小飞哥,加入 电报粉丝群 获得最新视频更新和问题解答。
后端面试系列将剖析后端面试中常考技术点,用尽量短的篇幅把一个一个技术点呈现出来。
shutdown()函数可以选择关闭全双工连接的读通道或者写通道,如果两个通道同时关闭,则这个连接不能再继续通信。close()函数会同时关闭全双工连接的读写通道,除了关闭连接外,还会释放套接字占用的文件描述符。而shutdown()只会关闭连接,但是不会释放占用的文件描述符。所以即使使用了SHUT_RDWR类型调用shutdown()关闭连接,也仍然要调用close()来释放连接占用的文件描述符。
close一个套接字的默认行为是把套接字标记为已关闭,然后立即返回到调用进程,该套接字描述符不能再由调用进程使用,然而TCP将尝试发送已经排队等待发送到对端的任何数据,发送完毕后发生的是正常的TCP连接终止序列。
该函数行为依赖how的值。
1.SHUT_RD:值为0,关闭连接的读这一半,套接字中不再有数据接收,且套接字接收缓冲区中的现有数据全都被丢弃,该套接字描述符不能再被进程调用,对端发送的数据会被确认,然后丢弃。
2.SHUT_WR:值为1,关闭连接的写这一半。这称为半关闭,当前在套接字发送缓冲区数据被发送,然后连接终止序列。不论套接字描述符引用技术是否等于0,写半部都会被关闭。
3.SHUT_RDWR:值为2,连接的读和写都关闭。相当于先调用SHUT_RD,再调用SHUT_WR。
close与shutdown的区别:
①:close函数函数会关闭套接字,如果由其他进程共享着这个套接字,那么它仍然是打开的,这个连接仍然可以用来读和写。
②:shutdown会切断进程共享的套接字的所有连接,不管引用计数是否为0,由第二个参数选择断连的方式。
终止网络连接的通常方法是调用close函数。不过close有两个限制,却可以使用shutdown来避免。
1 close把描述字的引用计数减1,仅在该计数变为0的时候才关闭套接口。而使用shutdown可以不管引用计数的值是多少就激发TCP的正常连接终止序列,也即是发送FIN节。
2 close终止数据传送的两个方向:读和写。而有的时候只是想关闭读或写,那么此时就使用shutdown函数进行关闭套接口描述字某一方向的操作。
在有父子进程的服务器程序中,套接口描述字是在父子进程之间共享的,因此它的引用计数为2。要是父进程调用close,那么这只是把该引用计数由2减为1,而且既然它仍然大于,FIN就不发送。这就是为什么在shutdown函数的原因,不管套接口的计数值为多少,FIN都必须被近发送出去。 当shutdown函数中的第2个参数为SHUT_WR 的时候,称为半关闭,此操作后,当前留在套接口发送缓冲中的数据将被发送掉,后跟TCP的正常连接终止序列。
现在总结一下shutdown()和close()的主要区别: 1)对应的系统调用不同 2)shutdown()只能用于套接字文件,close()可以用于所有文件类型 3)shutdown()只是关闭连接,并没有释放文件描述符,close()可以 4)shutdown()不能用于TCP_CLOSE状态的套接字,否则会返回 ENOTCONN 错误 5)shutdown()可以选择关闭读通道或写通道,close()不能。