Socket 編程IO Multiplexing

   Linux Socket 編程中I/O Multiplexing 主要經過三個函數來實現:select, poll,epoll來實現。I/O Multiplexing,先構造一張有關描述符的列表,而後調用一個函數,直到這些描述符中的一個已準備好進行I/O時,該函數才返回。在返回時,它告訴進程哪些描述符已準備好能夠進行I/O。本文具體介紹一下select 和poll的用法,給出簡單的demo代碼,簡要分析一下這兩個函數的使用易出錯的地方。        html

#include<sys/select.h>

int select(int maxfdp1, fd_set *restrict readfds, fd_set *restrict writefds,fd_set *restrict exceptfds, struct timeval* restrict tvptr);

//Returns: count of ready descriptors, 0 on timeout, -1 on error

  中間三個參數readfds、writefds和exceptfds是指向描述符集的指針。這三個描述符集說明了咱們關心的可讀、可寫或出於異常條件的各個描述符,設置爲NULL則表示不關心。每一個描述符集存放在一個fd_set數據類型中。這種數據類型爲每一可能的描述符保持一位。描述符集的函數接口(可能實現爲宏)包括:調用FD_ZERO將一個指定的fd_set變量的全部位設置爲0;調用FD_SET設置一個fd_set變量的指定位;調用FD_CLR將一指定位清楚;調用FD_ISSET測試一指定位是否設置。編程

#include <sys/select.h>

int FD_ISSET(int fd, fd_set *fdset);

  //Returns: nonzero if fd is in set, 0 otherwise

void FD_CLR(int fd, fd_set *fdset);

void FD_SET(int fd, fd_set *fdset);

void FD_ZERO(fd_set *fdset);
  

 

  文件描述符集fdset中的文件描述符的個數是有限制的,最大值由FD_SETSIZE指定,通常爲1024.網絡

  Select 最後一個參數用於設置超時值,當select監聽達到超時值時還未有關心的事件發生則返回,函數返回值爲0.socket

struct timeval{

  long tv_sec;//second

  long tv_usec;//microsecond

}

 

  超時參數若是設置爲 NULL 則無限等待。函數

  下面來是一個簡單的select Echo server:工具

// simpleEcho.cpp
#include <stdio.h> #include <sys/socket.h> #include <netinet/in.h> #include <unistd.h> #include <arpa/inet.h> #include <sys/types.h> #include <vector> #include <string.h> #include <stdlib.h> #include <fcntl.h> #define SEVER_PORT 1314 #define MAX_LINE_LEN 1024 using namespace std; int main() { struct sockaddr_in cli_addr, server_addr; socklen_t sock_len; vector<int> client(FD_SETSIZE,-1); fd_set rset,allset; int listenfd, connfd, sockfd, maxfd, nready, ix,maxid, nrcv,one; char addr_str[INET_ADDRSTRLEN],buf[MAX_LINE_LEN]; bzero(&server_addr,sizeof server_addr); server_addr.sin_family = AF_INET; server_addr.sin_addr.s_addr = htonl(INADDR_ANY); server_addr.sin_port = htons(SEVER_PORT); listenfd = socket(AF_INET,SOCK_STREAM,0); one = 1; setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR,&one, sizeof(one)); if(bind (listenfd ,(struct sockaddr *)&server_addr ,sizeof server_addr) < 0 ) { printf("socket bind error" ); return 0; } listen(listenfd ,10); FD_ZERO(&allset); FD_SET(listenfd ,&allset ); maxfd = listenfd ; maxid = -1 ; while(1 ) { rset = allset; //! nready = select (maxfd + 1, &rset,NULL,NULL,NULL); if(nready < 0 ) { printf("select error! \n" ); exit(1 ); } if(FD_ISSET (listenfd , &rset )) { sock_len = sizeof cli_addr; connfd = accept (listenfd ,(struct sockaddr *)&cli_addr , &sock_len); printf("recieve from : %s at port %d\n" , inet_ntop(AF_INET,&cli_addr .sin_addr ,addr_str ,INET_ADDRSTRLEN ),cli_addr .sin_port ); for(ix = 0 ; ix < static_cast< int>(client .size()); ix++) { if(client[ix] < 0 ) { client[ix] = connfd ; break; } } printf("client[%d] = %d\n" ,ix ,connfd ); if( FD_SETSIZE == ix) { printf("too many client! \n" ); exit(1 ); } if( connfd > maxfd) { maxfd = connfd; } FD_SET(connfd, &allset ); if(ix > maxid ) { maxid = ix; } if(--nready == 0) { continue; } } for(ix = 0 ; ix <= maxid; ix++) //<= { if((sockfd = client [ix ]) < 0) { continue; } if(FD_ISSET (sockfd ,&rset )) { if( 0 == (nrcv = read(sockfd,buf,MAX_LINE_LEN ))) { close(sockfd); client[ix] = -1 ; FD_CLR(sockfd ,&allset ); } else { printf("RECIEVE: %s \n" ,buf ); write(sockfd,buf,nrcv); } } if(--nready == 0) { break; } } } return 0; }

  在使用select 的時候要注意兩點:測試

    第一個參數須要是當前所關心的文件描述符中最大的一個+1spa

    第二須要注意的是select的中間3個參數採用了「value-result」(UNP1的說法)的方式,設置了關心的文件描述符進行select,select返回以後對應描述的fdset中只有有事件發生的對應fd會被設置,其它關心可是沒有事件發生的描述符將會從fdset中清除掉,若是不進行從新賦值,下次select就不會關注這些描述符了,所以上述代碼中allset每次對rset進行復制。.net

   來看看若是隻在while(1) 以前設置rset,在while(1) 中不在每次select以前賦值會發生什麼,在控制檯輸入: strace ./simpleEcho,另外打開一個控制檯窗口輸入:nc localhost 1314,這做爲一個鏈接Echo server 的 client,而後輸入你想發往Echo Server內容。關鍵咱們來看一下Echo server的狀況:指針

  能夠看到 select 首先關注的文件描述符 fd == 3,該描述符是listenfd,而後有client連過來,select關注了 fd 3 和 4,4是accept函數打開的用於與client通訊的描述符,當client向server寫數據以後select關注的描述就只剩下 fd 4了,也就是當前處於鏈接狀態的描述符,若是client主動關閉,select返回以後,下次監聽就沒有關注的描述符了,可見select函數的「value-result」 返回方式是這樣工做的:每次只返回監聽描述符中處於active的,其它處於監聽的可是當前沒有事件發生的描述符則會從監聽的fdset中清除掉。所以在每次select以前須要給關注的fdset從新賦值。

  注1:在進行系統調用調試的時候 strace 是一個利器,簡單使用方式如上面在運行程序以前加上 strace 便可。在調試代碼邏輯的時候固然仍是使用gdb了。

  注2Netcat 或者叫 nc 是 Linux 下的一個用於調試和檢查網絡工具包。可用於建立 TCP/IP 鏈接,最大的用途就是用來處理 TCP/UDP 套接字。

  

  select 何時會處於準備好並返回呢? UNPv1 上進行了詳細介紹:

  下面四個條件任何一個知足的時候套件字準備好讀:

  1. 套接口接受緩衝區的數據字節數大於等於套接口接受緩衝區的低潮限度當前值。對這樣的套接口讀操做將不阻塞並返回一個大於0的值(既準備好讀入的數據量)。咱們能夠用套接口選項SO_RCVLOWAT來設置此低潮限度,對於TCP和UDP套接口,其缺省值爲1。

  2. 鏈接的讀這一半關閉(也就是接收了FIN的TCP鏈接)。對這樣的套接口讀操做將不阻塞並返回0(記文件結束符)。

  3. 套接口是一個監聽的套接口且已完成的鏈接數爲非0。正常狀況下這樣的套接口上的accpet不會被阻塞。

  4. 有一個套接口錯誤待處理。對這樣的套接口操做將不阻塞並返回一個錯誤-1,errno設置成明確的錯誤條件。

 

  如下三個條件的任何一個知足時,套接口準備好寫操做:

  1. 套接口發送緩衝區中可用空間的字節數大於等於套接口發送緩衝區低潮限度的當前值,且或者(i)套接口已鏈接,或者(ii)套接口不須要鏈接(例如UDP套接字)。這意味着,若是咱們將這樣的套接口設置爲非阻塞,寫操做將不阻塞且返回一個正值(例如由傳輸層傳入的字節數)。咱們能夠用套接口選項SO_SNDLOWAT來設置此低潮限度,對於TCP和UDP套接口其缺省值爲2048.

  2. 鏈接的寫這一半關閉,對這樣的套接口寫操做將產生信號SIGPIPE。

  3. 有一個套接口錯誤待處理。對這樣的套接口操做寫操做將不阻塞且返回一個錯誤-1,errno設置成明確的錯誤條件。這些待處理的錯誤也可經過指定套接口選項SO_ERROR調用getsockopt來取得並清除。

  

  若是一個套接口存在帶外數據或者仍處於帶外標記,那他有異常條件待處理。

   poll留到下一篇吧……

  但,I/O multiplexing 就是這樣用的嗎?

相關文章
相關標籤/搜索