[zz] Linux socket關閉連接shutdown與close

http://www.cublog.cn/u3/99348/showart_2072194.html

 

在Linux socket關閉連接的方法有兩種分別是shutdown和close,首先看一下shutdown的定義

 

#include<sys/socket.h>

int shutdown(int sockfd,int how);

how的方式有三種分別是

SHUT_RD(0):關閉sockfd上的讀功能,此選項將不允許sockfd進行讀操作。

SHUT_WR(1):關閉sockfd的寫功能,此選項將不允許sockfd進行寫操作。

SHUT_RDWR(2):關閉sockfd的讀寫功能。

成功則返回0,錯誤返回-1,錯誤碼errno:EBADF表示sockfd不是一個有效描述符;ENOTCONN表示sockfd未連接;ENOTSOCK表示sockfd是一個文件描述符而不是socket描述符。

 

close的定義如下:

#include<unistd.h>

int close(int fd);

關閉讀寫。

成功則返回0,錯誤返回-1,錯誤碼errno:EBADF表示fd不是一個有效描述符;EINTR表示close函數被信號中斷;EIO表示一個IO錯誤。

 

 

下面摘用網上的一段話來說明二者的區別:

close--------關閉本進程的socket id,但鏈接還是開着的,用這個socket id的其它進程還能用這個鏈接,能讀或寫這個socket id
shutdown--則破壞了socket 鏈接,讀的時候可能偵探到EOF結束符,寫的時候可能會收到一個SIGPIPE信號,這個信號可能直到

socket buffer被填充了才收到,shutdown還有一個關閉方式的參數,0 不能再讀,1不能再寫,2 讀寫都不能。

socket 多進程中的shutdown, close使用
當所有的數據操作結束以後,你可以調用close()函數來釋放該socket,從而停止在該socket上的任何數據操作:
close(sockfd);


你也可以調用shutdown()函數來關閉該socket。該函數允許你只停止在某個方向上的數據傳輸,而一個方向上的數據傳輸繼

續進行。如你可以關閉某socket的寫操作而允許繼續在該socket上接受數據,直至讀入所有數據。
int shutdown(int sockfd,int how);
Sockfd是需要關閉的socket的描述符。參數 how允許爲shutdown操作選擇以下幾種方式:
SHUT_RD:關閉連接的讀端。也就是該套接字不再接受數據,任何當前在套接字接受緩衝區的數據將被丟棄。進程將不能對該

套接字發出任何讀操作。對TCP套接字該調用之後接受到的任何數據將被確認然後無聲的丟棄掉。
SHUT_WR:關閉連接的寫端,進程不能在對此套接字發出寫操作
SHUT_RDWR:相當於調用shutdown兩次:首先是以SHUT_RD,然後以SHUT_WR


使用close中止一個連接,但它只是減少描述符的參考數,並不直接關閉連接,只有當描述符的參考數爲0時才關閉連接。
shutdown可直接關閉描述符,不考慮描述符的參考數,可選擇中止一個方向的連接。

注意:


1>. 如果有多個進程共享一個套接字,close每被調用一次,計數減1,直到計數爲0時,也就是所用進程都調用了close,套接字將被釋放。


2>. 在多進程中如果一個進程中shutdown(sfd, SHUT_RDWR)後其它的進程將無法進行通信. 如果一個進程close(sfd)將不會影響到其它進程. 得自己理解引用計數的用法了. 有Kernel編程知識的更好理解了.

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章