linux為什么要用select

因為select可以使開發者在同時等待多個文件緩沖區,可減少IO等待的時間,能夠提高進程的IO效率。select()函數是IO多路復用的函數,允許程序監視多個文件描述符,等待所監視的一個或者多個文件描述符變為“準備好”的狀態;所謂的”準備好“狀態是指:文件描述符不再是阻塞狀態,可以用于某類IO操作了,包括可讀,可寫,發生異常三種。

linux為什么要用select

本教程操作環境:linux7.3系統、Dell G3電腦。

select是一個計算機函數,位于頭文件#include 。該函數用于監視文件描述符的變化情況——讀寫或是異常。

1. select函數介紹

select函數是IO多路復用的函數,它主要的功能是用來等文件描述符中的事件是否就緒,select可以使我們在同時等待多個文件緩沖區 ,減少IO等待的時間,能夠提高進程的IO效率。

select()函數允許程序監視多個文件描述符,等待所監視的一個或者多個文件描述符變為“準備好”的狀態。所謂的”準備好“狀態是指:文件描述符不再是阻塞狀態,可以用于某類IO操作了,包括可讀,可寫,發生異常三種

2. select函數參數的介紹

???????int?select(int?nfds,?fd_set?*readfds,?fd_set?*writefds, ??????????????????fd_set?*exceptfds,?Struct?timeval?*timeout);

ndfs

等待的文件描述符的最大值+1,例如:應用進程想要去等待文件描述符3,5,8的事件,則

nfds=max(3,5,8)+1;

fd_set類型

readfds和writefds,exceptfds的類型都是fd_set,那么fd_set類型是什么呢?

  • fd_set類型本質是一個位圖位圖的位置 表示 相對應的文件描述符,內容表示該文件描述符是否有效,1代表該位置的文件描述符有效,0則表示該位置的文件描述符無效。
  • 如果將文件描述符2,3設置位圖當中,則位圖表示的是為1100。
  • fd_set的上限是1024個文件描述符。

readfds

  • readfds是 等待讀事件的文件描述符集合,.如果不關心讀事件(緩沖區有數據),則可以傳NULL值。
  • 應用進程和內核都可以設置readfds,應用進程設置readfds是為了通知內核去等待readfds中的文件描述符的讀事件.而 內核設置readfds是為了告訴應用進程哪些讀事件生效

linux為什么要用select

writefds

與readfds類似,writefds是等待寫事件(緩沖區中是否有空間)的集合,如果不關心寫事件,則可以傳值NULL。

exceptfds

如果內核等待相應的文件描述符發生異常,則將失敗的文件描述符設置進exceptfds中,如果不關心錯誤事件,可以傳值NULL。

timeout

設置select在內核中阻塞的時間,如果想要設置為非阻塞,則設置為NULL。如果想讓select阻塞5秒,則將創建一個struct timeval time={5,0};

其中struct timeval的結構體類型是:

???????????struct?timeval?{ ???????????????long????tv_sec;?????????/*?seconds?*/ ???????????????long????tv_usec;????????/*?microseconds?*/ ???????????};

返回值

  • 如果沒有文件描述符就緒就返回0;
  • 如果調用失敗返回-1;
  • 如果timeout中中readfds中有事件發生,則返回timeout剩下的時間。

3.select的工作流程

應用進程內核都需要從readfds和writefds獲取信息,其中,內核需要從readfds和writefds知道哪些文件描述符需要等待,應用進程需要從readfds和writefds中知道哪些文件描述符的事件就緒.

linux為什么要用select

如果我們要不斷輪詢等待文件描述符,則應用進程需要不斷的重新設置readfds和writefds,因為每一次調用select,內核會修改readfds和writefds,所以我們需要在 應用程序設置一個數組 來保存程序需要等待的文件描述符,保證調用 select 的時候readfds 和 writefds中的將如下:

linux為什么要用select

4.Select服務器

?如果是一個select服務器進程,則服務器進程會不斷的接收有新鏈接每個鏈接對應一個文件描述符,如果想要我們的服務器能夠同時等待多個鏈接的數據的到來,我們監聽套接字listen_sock讀取新鏈接的時候,我們需要將新鏈接的文件描述符保存到read_arrys數組中,下次輪詢檢測的就會將新鏈接的文件描述符設置進readfds中,如果有鏈接關閉,則將相對應的文件描述符從read_arrys數組中拿走

一張圖看懂select服務器:

linux為什么要用select

簡易版的select服務器:

server.hpp文件:

#pragma?once?? ??#include<iostream> ??#include<sys> ??#include<sys>???? ??#include<netinet>? ??#include<string.h> ??using?std::cout; ??using?std::endl; ??#define?BACKLOG?5?? ?????? ??namespace?sjp{???? ????class?server{???? ??????public:???? ??????static?int?Socket(){???? ????????int?sock=socket(AF_INET,SOCK_STREAM,0);???? ????????if(sock&gt;0)???? ????????return?sock;???? ????????if(sock<p>?select_server.hpp文件</p> <pre class="brush:js;toolbar:false;">#pragma?once ??#include<vector> ??#include"server.hpp" ??#include<unistd.h> ??#include<time.h> ?? ??namespace?Select{ ????class?select_server{ ??????private: ????????int?listen_sock;//監聽套接字???? ????????int?port;???? ?????????? ??????public:???? ????????select_server(int?_port):port(_port){}???? ?????? ????????//初始化select_server服務器???? ????????void?InitServer(){???? ??????????listen_sock=sjp::server::Socket();???? ??????????sjp::server::Bind(listen_sock,port);???? ??????????sjp::server::Listen(listen_sock);???? ????????}???? ?????? ?????? ????????void?Run(){???? ??????????std::vector<int>?readfds_arry(1024,-1);//readfds_arry保存讀事件的文件描述符???? ??????????readfds_arry[0]=listen_sock;//將監聽套接字保存進readfds_arry數組中???? ??????????fd_set?readfds;???? ??????????while(1){???? ??????????FD_ZERO(&amp;readfds);???? ??????????int?nfds=0;???? ??????????//將read_arry數組中的文件描述符設置進程readfds_arry位圖中???? ??????????for(int?i=0;i&amp;?readfds_arry,fd_set?readfds){ ????????for(int?i=0;i<readfds_arry.size>&amp;?fds_arry,int?fd){ ???????for(int?i=0;i<fds_arry.size><p>?select_server.cc文件</p> <pre class="brush:js;toolbar:false;">#include"select_server.hpp"  int?main(int?argv,char*?argc[]){ ??if(argv!=2){ ????coutInitServer(); ??sl-&gt;Run(); }

測試:

linux為什么要用select

linux為什么要用select

5.Select的缺陷

  • 由于fd_set的上限是1024,所以select能等待的讀事件的文件描述符和寫事件的文件描述是有上限的,如果作為一個大型服務器,能夠同時鏈接的客戶端是遠遠不夠的。
  • 每次應用進程調用一次select之前,都需要重新設定writefds和readfds,如果進行輪詢調用select,這對影響cpu效率。
  • 內核每一次等待文件描述符 都會重新掃描所有readfds或者writefds中的所有文件描述符,如果有較多的文件描述符,則會影響效率。

推薦學習:Linux視頻教程

以上就是

? 版權聲明
THE END
喜歡就支持一下吧
點贊8 分享