|簡體中文

比思論壇

 找回密碼
 按這成為會員
搜索



查看: 438|回復: 2
打印 上一主題 下一主題

BSD Socket的使用

[複製鏈接]

25

主題

0

好友

144

積分

小學生

Rank: 2

  • TA的每日心情

    2015-9-9 16:11
  • 簽到天數: 1 天

    [LV.1]初來乍到

    推廣值
    0
    貢獻值
    0
    金錢
    1739
    威望
    144
    主題
    25
    樓主
    發表於 2015-9-9 16:28:13
    recv和send
    recv和send函数提供了和read和write差不多的功能.不过它们提供 了第四个参数来控制读写操作.

    int recv(int sockfd,void *buf,int len,int flags)
    int send(int sockfd,void *buf,int len,int flags)

    前面的三个参数和read,write一样,第四个参数可以是0或者是以下的组合
    _______________________________________________________________
    | MSG_DONTROUTE | 不查找路由表 |
    | MSG_OOB | 接受或者发送带外数据 |
    | MSG_PEEK | 查看数据,并不从系统缓冲区移走数据 |
    | MSG_WAITALL | 等待所有数据 |
    |--------------------------------------------------------------|

    MSG_DONTROUTE:是send函数使用的标志.这个标志告诉IP协议.目的主机在本地网络上面,没有必要查找路由表.这个标志一般用网络诊断和路由程序里面.
    MSG_OOB:表示可以接收和发送带外的数据.关于带外数据我们以后会解释的.

    MSG_PEEK:是recv函数的使用标志,表示只是从系统缓冲区中读取内容,而不清楚系统缓冲区的内容.这样下次读的时候,仍然是一样的内容.一般在有多个进程读写数据时可以使用这个标志.

    MSG_WAITALL是recv函数的使用标志,表示等到所有的信息到达时才返回.使用这个标志的时候recv回一直阻塞,直到指定的条件满足,或者是发生了错误. 1)当读到了指定的字节时,函数正常返回.返回值等于len 2)当读到了文件的结尾时,函数正常返回.返回值小于len 3)当操作发生错误时,返回-1,且设置错误为相应的错误号(errno)

    如果flags为0,则和read,write一样的操作.还有其它的几个选项,不过我们实际上用的很少,可以查看 Linux Programmer's Manual得到详细解释.


    在UNIX/LINUX下,非阻塞模式SOCKET可以采用recv+MSG_PEEK的方式进行判断,其中MSG_PEEK保证了仅仅进行状态判断,而不影响数据接收
    对于主动关闭的SOCKET, recv返回-1,而且errno被置为9(#define EBADF   9 /* Bad file number */)
    或104 (#define ECONNRESET 104 /* Connection reset by peer */)
    对于被动关闭的SOCKET,recv返回0,而且errno被置为11(#define EWOULDBLOCK EAGAIN /* Operation would block */)
    对正常的SOCKET, 如果有接收数据,则返回>0, 否则返回-1,而且errno被置为11(#define EWOULDBLOCK EAGAIN /* Operation would block */)
    因此对于简单的状态判断(不过多考虑异常情况),
        recv返回>0,   正常
        返回-1,而且errno被置为11  正常
        其它情况    关闭
    注意要区分不同操作系统分别进行测试, 包括WINDOWS, LINUX和UNIX会各有不同
    在WINDOWS下比较简单,可以使用FD_CLOSE事件判断SOCKET是否已经断开

    view plaincopy to clipboardprint?
    01.bool IsSocketClosed(SOCKET clientSocket)  
    02.{  
    03. bool ret = false;  
    04. HANDLE closeEvent = WSACreateEvent();  
    05. WSAEventSelect(clientSocket, closeEvent, FD_CLOSE);  
    06.  
    07. DWORD dwRet = WaitForSingleObject(closeEvent, 0);  
    08.   
    09. if(dwRet == WSA_WAIT_EVENT_0)  
    10.  ret = true;  
    11. else if(dwRet == WSA_WAIT_TIMEOUT)  
    12.  ret = false;  
    13.  
    14. WSACloseEvent(closeEvent);  
    15. return ret;  
    16.}

    0

    主題

    0

    好友

    111

    積分

    小學生

    Rank: 2

  • TA的每日心情
    擦汗
    2024-3-10 11:38
  • 簽到天數: 80 天

    [LV.6]常住居民II

    推廣值
    0
    貢獻值
    0
    金錢
    661
    威望
    111
    主題
    0
    沙發
    發表於 2016-3-25 15:02:45
    先赞一个
    重要聲明:本論壇是以即時上載留言的方式運作,比思論壇對所有留言的真實性、完整性及立場等,不負任何法律責任。而一切留言之言論只代表留言者個人意見,並非本網站之立場,讀者及用戶不應信賴內容,並應自行判斷內容之真實性。於有關情形下,讀者及用戶應尋求專業意見(如涉及醫療、法律或投資等問題)。 由於本論壇受到「即時上載留言」運作方式所規限,故不能完全監察所有留言,若讀者及用戶發現有留言出現問題,請聯絡我們比思論壇有權刪除任何留言及拒絕任何人士上載留言 (刪除前或不會作事先警告及通知 ),同時亦有不刪除留言的權利,如有任何爭議,管理員擁有最終的詮釋權。用戶切勿撰寫粗言穢語、誹謗、渲染色情暴力或人身攻擊的言論,敬請自律。本網站保留一切法律權利。

    手機版| 廣告聯繫

    GMT+8, 2024-5-23 17:21 , Processed in 0.013392 second(s), 16 queries , Gzip On, Memcache On.

    Powered by Discuz! X2.5

    © 2001-2012 Comsenz Inc.

    回頂部