Linux中select poll和epoll的區別

在Linux Socket服務器短編程時,爲了處理大量客戶的鏈接請求,須要使用非阻塞I/O和複用,select、poll和epoll是Linux API提供的I/O複用方式,自從Linux 2.6中加入了epoll以後,在高性能服務器領域獲得普遍的應用,如今比較出名的nginx就是使用epoll來實現I/O複用支持高併發,目前在高並 發的場景下,nginx愈來愈收到歡迎。這裏有個文章參考。Nginx成爲全球Top1000網站最受歡迎的Web服務器。linux

據 w3techs 7月 3 日的統計數據代表,在全球 Top 1000 的網站中,有 34.9% 的網站在使用 Nginx,這使得 Nginx 超越了 Apache,成爲了高流量網站最信任的 Web 服務器。下圖是統計數據。nginx

select:

下面是select的函數接口:編程

int select (int n, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);

 

select 函數監視的文件描述符分3類,分別是writefds、readfds、和exceptfds。調用後select函數會阻塞,直到有描述副就緒(有數據 可讀、可寫、或者有except),或者超時(timeout指定等待時間,若是當即返回設爲null便可),函數返回。當select函數返回後,能夠 經過遍歷fdset,來找到就緒的描述符。服務器

select目前幾乎在全部的平臺上支持,其良好跨平臺支持也是它的一個優勢。select的一 個缺點在於單個進程可以監視的文件描述符的數量存在最大限制,在Linux上通常爲1024,能夠經過修改宏定義甚至從新編譯內核的方式提高這一限制,但 是這樣也會形成效率的下降。數據結構

poll:

int poll (struct pollfd *fds, unsigned int nfds, int timeout);

 

不一樣與select使用三個位圖來表示三個fdset的方式,poll使用一個 pollfd的指針實現。併發

struct pollfd {int fd; /* file descriptor */short events; /* requested events to watch */short revents; /* returned events witnessed */};

 

pollfd結構包含了要監視的event和發生的event,再也不使用select「參數-值」傳遞的方式。同時,pollfd並無最大數量限制(可是數量過大後性能也是會降低)。 和select函數同樣,poll返回後,須要輪詢pollfd來獲取就緒的描述符。socket

從上面看,select和poll都須要在返回後,經過遍歷文件描述符來獲取已經就緒的socket。事實上,同時鏈接的大量客戶端在一時刻可能只有不多的處於就緒狀態,所以隨着監視的描述符數量的增加,其效率也會線性降低。函數

epoll:

epoll的接口以下:高併發

複製代碼

int epoll_create(int size);int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
            typedef union epoll_data {                void *ptr;                int fd;
                __uint32_t u32;
                __uint64_t u64;
            } epoll_data_t;            struct epoll_event {
                __uint32_t events;      /* Epoll events */
                epoll_data_t data;      /* User data variable */
            };int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout);

複製代碼

 

主要是epoll_create,epoll_ctl和epoll_wait三個函數。epoll_create函數建立epoll文件描述符,參數size並非限制了epoll所能監聽的描述符最大個數,只是對內核初始分配內部數據結構的一個建議。返回是epoll描述符。-1表示建立失敗。epoll_ctl 控制對指定描述符fd執行op操做,event是與fd關聯的監聽事件。op操做有三種:添加EPOLL_CTL_ADD,刪除EPOLL_CTL_DEL,修改EPOLL_CTL_MOD。分別添加、刪除和修改對fd的監聽事件。epoll_wait 等待epfd上的io事件,最多返回maxevents個事件。性能

在 select/poll中,進程只有在調用必定的方法後,內核纔對全部監視的文件描述符進行掃描,而epoll事先經過epoll_ctl()來註冊一 個文件描述符,一旦基於某個文件描述符就緒時,內核會採用相似callback的回調機制,迅速激活這個文件描述符,當進程調用epoll_wait() 時便獲得通知。

epoll的優勢主要是一下幾個方面:

1. 監視的描述符數量不受限制,它所支持的FD上限是最大能夠打開文件的數目,這個數字通常遠大於2048,舉個例子,在1GB內存的機器上大約是10萬左 右,具體數目能夠cat /proc/sys/fs/file-max察看,通常來講這個數目和系統內存關係很大。select的最大缺點就是進程打開的fd是有數量限制的。這對 於鏈接數量比較大的服務器來講根本不能知足。雖然也能夠選擇多進程的解決方案( Apache就是這樣實現的),不過雖然linux上面建立進程的代價比較小,但仍舊是不可忽視的,加上進程間數據同步遠比不上線程間同步的高效,因此也 不是一種完美的方案。

2. IO的效率不會隨着監視fd的數量的增加而降低。epoll不一樣於select和poll輪詢的方式,而是經過每一個fd定義的回調函數來實現的。只有就緒的fd纔會執行回調函數。

3.支持電平觸發和邊沿觸發(只告訴進程哪些文件描述符剛剛變爲就緒狀態,它只說一遍,若是咱們沒有采起行動,那麼它將不會再次告知,這種方式稱爲邊緣觸發)兩種方式,理論上邊緣觸發的性能要更高一些,可是代碼實現至關複雜。

4.mmap加速內核與用戶空間的信息傳遞。epoll是經過內核於用戶空間mmap同一塊內存,避免了無畏的內存拷貝。

相關文章
相關標籤/搜索