徹底學會使用epoll(六)——關於ET的若干問題總結
徹底學會使用epoll(六)——關於ET的若干問題總結
6.1 ET模式為什麼要設定在非阻塞模式下工作
因為ET模式下的讀寫需要一直讀或寫直到出錯(對於讀,當讀到的實際位元組數小於請求位元組數時就可以停止),而如果你的檔案描述符如果不是非阻塞的,那這個一直讀或一直寫勢必會在最後一次阻塞。這樣就不能在阻塞在epoll_wait上了,造成其他檔案描述符的任務餓死。
6.2 使用ET和LT的區別
LT:水平觸發,效率會低於ET觸發,尤其在大併發,大流量的情況下。但是LT對程式碼編寫要求比較低,不容易出現問題。LT模式服務編寫上的表現是:只要有資料沒有被獲取,核心就不斷通知你,因此不用擔心事件丟失的情況。
ET:邊緣觸發,效率非常高,在併發,大流量的情況下,會比LT少很多epoll的系統呼叫,因此效率高。但是對程式設計要求高,需要細緻的處理每個請求,否則容易發生丟失事件的情況。
下面舉一個列子來說明LT和ET的區別(都是非阻塞模式,阻塞就不說了,效率太低):
採用LT模式下,如果accept呼叫有返回就可以馬上建立當前這個連線了,再epoll_wait等待下次通知,和select一樣。
但是對於ET而言,如果accpet呼叫有返回,除了建立當前這個連線外,不能馬上就epoll_wait還需要繼續迴圈accpet,直到返回-1,且errno==EAGAIN,
從本質上講:與LT相比,ET模型是通過減少系統呼叫來達到提高並行效率的。
6.3 一道騰訊後臺開發的面試題
使用Linux epoll模型,水平(LT)觸發模式,當socket可寫時,會不停的觸發socket可寫的事件,如何處理?
第一種最普遍的方式:
需要向socket寫資料的時候才把socket加入epoll,等待可寫事件。接受到可寫事件後,呼叫write或者send傳送資料。當所有資料都寫完後,把socket移出epoll。
這種方式的缺點是,即使傳送很少的資料,也要把socket加入epoll,寫完後在移出epoll,有一定操作代價。
一種改進的方式:
開始不把socket加入epoll,需要向socket寫資料的時候,直接呼叫write或者send傳送資料。如果返回EAGAIN,把socket加入epoll,在epoll的驅動下寫資料,全部資料傳送完畢後,再移出epoll。
這種方式的優點是:資料不多的時候可以避免epoll的事件處理,提高效率。
6.4什麼情況下用ET
很簡單,當你想提高程式效率的時候。
最後附一個epoll例項:
點選(此處)摺疊或開啟
-
#include <sys/socket.h>
-
#include <sys/wait.h>
-
#include <netinet/in.h>
-
#include <netinet/tcp.h>
-
#include <sys/epoll.h>
-
#include <sys/sendfile.h>
-
#include <sys/stat.h>
-
#include <unistd.h>
-
#include <stdio.h>
-
#include <stdlib.h>
-
#include <string.h>
-
#include <strings.h>
-
#include <fcntl.h>
-
#include <errno.h>
-
-
#define MAX_EVENTS 10
-
#define PORT 8080
-
-
//設定socket連線為非阻塞模式
-
void setnonblocking(int sockfd) {
-
int opts;
-
opts = fcntl(sockfd, F_GETFL);
-
if(opts < 0) {
-
perror("fcntl(F_GETFL)\n");
-
exit(1);
-
}
-
opts = (opts | O_NONBLOCK);
-
if(fcntl(sockfd, F_SETFL, opts) < 0) {
-
perror("fcntl(F_SETFL)\n");
-
exit(1);
-
}
-
}
-
-
int main(){
-
struct epoll_event ev, events[MAX_EVENTS]; //ev負責新增事件,events接收返回事件
-
int addrlen, listenfd, conn_sock, nfds, epfd, fd, i, nread, n;
-
struct sockaddr_in local, remote;
-
char buf[BUFSIZ];
-
-
//建立listen socket
-
if( (listenfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
-
perror("sockfd\n");
-
exit(1);
-
}
-
setnonblocking(listenfd);//listenfd設定為非阻塞[1]
-
bzero(&local, sizeof(local));
-
local.sin_family = AF_INET;
-
local.sin_addr.s_addr = htonl(INADDR_ANY);;
-
local.sin_port = htons(PORT);
-
if( bind(listenfd, (struct
sockaddr *) &local, sizeof(local)) < 0) {
-
perror("bind\n");
-
exit(1);
-
}
-
listen(listenfd, 20);
-
-
epfd = epoll_create(MAX_EVENTS);
-
if (epfd == -1) {
-
perror("epoll_create");
-
exit(EXIT_FAILURE);
-
}
-
-
ev.events = EPOLLIN;
-
ev.data.fd = listenfd;
-
if (epoll_ctl(epfd, EPOLL_CTL_ADD, listenfd, &ev) == -1) {//監聽listenfd
-
perror("epoll_ctl: listen_sock");
-
exit(EXIT_FAILURE);
-
}
-
-
for (;;) {
-
nfds = epoll_wait(epfd, events, MAX_EVENTS, -1);
-
if (nfds == -1) {
-
perror("epoll_pwait");
-
exit(EXIT_FAILURE);
-
}
-
-
for (i = 0; i < nfds; ++i) {
-
fd = events[i].data.fd;
-
if (fd == listenfd) {
-
while ((conn_sock = accept(listenfd,(struct
sockaddr *) &remote,
-
(size_t *)&addrlen)) > 0) {
-
setnonblocking(conn_sock);//下面設定ET模式,所以要設定非阻塞
-
ev.events = EPOLLIN | EPOLLET;
-
ev.data.fd = conn_sock;
-
if (epoll_ctl(epfd, EPOLL_CTL_ADD, conn_sock, &ev) == -1) {//讀監聽
-
perror("epoll_ctl: add"); //連線套接字
-
exit(EXIT_FAILURE);
-
}
-
}
-
if (conn_sock == -1) {
-
if (errno != EAGAIN && errno != ECONNABORTED
-
&& errno != EPROTO && errno != EINTR)
-
perror("accept");
-
}
-
continue;
-
}
-
if (events[i].events & EPOLLIN) {
-
n = 0;
-
while ((nread = read(fd, buf + n, BUFSIZ-1)) > 0) {//ET下可以讀就一直讀
-
n += nread;
-
}
-
if (nread == -1 && errno != EAGAIN) {
-
perror("read error");
-
}
-
ev.data.fd = fd;
-
ev.events = events[i].events | EPOLLOUT; //MOD OUT
-
if (epoll_ctl(epfd, EPOLL_CTL_MOD, fd, &ev) == -1) {
-
perror("epoll_ctl: mod");
-
}
-
}
-
if (events[i].events & EPOLLOUT) {
-
sprintf(buf, "HTTP/1.1 200 OK\r\nContent-Length: %d\r\n\r\nHello
World", 11);
-
int nwrite, data_size = strlen(buf);
-
n = data_size;
-
while (n > 0) {
-
nwrite = write(fd, buf + data_size - n, n);//ET下一直將要寫資料寫完
-
if (nwrite < n) {
-
if (nwrite == -1 && errno != EAGAIN) {
-
perror("write error");
-
}
-
break;
-
}
-
n -= nwrite;
-
}
-
close(fd);
-
}
-
}
-
}
-
return 0;
- }
相關文章
- 徹底學會使用epoll(一)——ET模式實現分析模式
- 徹底學會使用epoll(三)——ET的讀操作例項分析
- 徹底學會使用epoll(四)——ET的寫操作例項分析
- 徹底學會使用epoll(五)—— ET模式下的注意事項模式
- 關於若干資料平均放入若干個盒子的問題
- 關於echarts使用的常見問題總結Echarts
- 徹底解決關於CSocket類的Receive超時的問題(轉)
- 關於move tablespace的問題總結
- 關於 多程式epoll 與 “驚群”問題
- 一行程式碼徹底終結關於物件/陣列的深拷貝問題行程物件陣列
- 關於SSM與echart結合的問題總結SSM
- 關於中文亂碼問題(總結)
- 關於修改分割槽表的問題總結
- C++ 關於static variables的學習中遇到的問題總結C++
- 終於徹底搞清楚了spin-lock 之一次CPU問題定位過程總結
- Java關於初始化問題的總結(一)Java
- 關於ora-02391問題的總結
- 關於釘釘直播回放影片下載若干方法的總結
- C++雜談之關於檔案操作的若干問題C++
- 關於單點登入的若干問題,請高手指教
- 【前端】一文徹底學會Promise前端Promise
- 在Centos和Docker上安裝STF 遇到的若干問題總結CentOSDocker
- 關於程式猿的六個問答題
- 這套方法論,徹底終結MySQL同步延遲問題MySql
- 徹底解決程式亂碼問題
- 「linux」例項淺析epoll的LT和ET模式,ET模式為何要使用非阻塞IOLinux模式
- 關於 flex 面試題總結Flex面試題
- 關於 JOIN 耐心總結,學不會你打我係列
- mysql相關問題總結MySql
- 關於exp/imp的總結學習
- Epoll在LT和ET模式下的讀寫方式模式
- 徹底解決VC6在編譯,連結時的假死問題編譯
- 16.徹底解決Jmap在mac版本無法使用的問題Mac
- 徹底解決Python編碼問題Python
- 徹底解決Hive小檔案問題Hive
- 徹底搞懂 python 中文亂碼問題Python
- 關於資料庫間連結問題彙總---Oracle資料庫Oracle
- 如何徹底解決pip install慢的問題