Linux編程之Epoll高併發

来源:https://www.cnblogs.com/znwang/archive/2018/07/08/9279834.html
-Advertisement-
Play Games

網路上所有資料都說epoll是高併發、單線程、IO重疊服用的首選架構,比select和poll性能都要好,特別是在有大量不活躍連接的情況下。具體原理就不闡述了,下麵說說使用。 具有有三個函數: #include <sys/epoll.h> 1、int epoll_create ( int size ...


網路上所有資料都說epoll是高併發、單線程、IO重疊服用的首選架構,比select和poll性能都要好,特別是在有大量不活躍連接的情況下。具體原理就不闡述了,下麵說說使用。

具有有三個函數:

#include <sys/epoll.h>

 

1、int epoll_create ( int size );

size是epoll要監視的fd的規模。

 

2、int epoll_ctl ( int epfd, int op, int fd, struct epoll_event *event );

(1)epfd:epoll_create的返回值。

(2)op  指定操作類型:

     EPOLL_CTL_ADD:往事件表中註冊fd上的事件

     EPOLL_CTL_MOD:修改fd上的註冊事件

     EPOLL_CTL_DEL:刪除fd上的註冊事件

(3)fd:要操作的文件描述符(socket)

 (4)event:指定要監聽fd的什麼事情。它是epoll_event結構指針類型:

struct epoll_event

{

     __unit32_t events;    // epoll事件

     epoll_data_t data;     // 用戶數據

};

events:描述事件類型。events可以是以下幾個巨集的集合:

 EPOLLIN :表示對應的文件描述符可以讀(包括對端SOCKET正常關閉);
EPOLLOUT:表示對應的文件描述符可以寫;

 EPOLLPRI:表示對應的文件描述符有緊急的數據可讀(這裡應該表示有帶外數據到來);
EPOLLERR:表示對應的文件描述符發生錯誤;

EPOLLHUP:表示對應的文件描述符被掛斷;

 EPOLLET: 將EPOLL設為邊緣觸發(Edge Triggered)模式,這是相對於水平觸發(Level Triggered)來說的。

EPOLLONESHOT:只監聽一次事件,當監聽完這次事件之後,如果還需要繼續監聽這個socket的話,需要再次把這個socket加入到EPOLL隊列里。

data成員:其中data.fd常用來裝要操作的fd。

 

typedef union epoll_data

{

     void *ptr;

      int fd;

      __uint32_t u32;

      __uint64_t u64;

  } epoll_data_t;

 

 

3、int epoll_wait ( int epfd, struct epoll_event* events, int maxevents, int timeout );

epoll_wait的工作流程是:等待,如果有epoll事件發生立刻返回,否則等待timeout毫秒。返回時拷貝要處理的事件到events指向的數組,返回就緒的文件描述符的個數,失敗時返回-1並設置errno。 

timeout:指定epoll的超時時間,單位是毫秒。當timeout為-1是,epoll_wait調用將永遠阻塞,直到某個事件發生。當timeout為0時,epoll_wait調用將立即返回。 

maxevents:指定最多監聽多少個事件。如果events指向的是20個單元的結構體數組,那麼就設置maxevents為20。 

events: events指向的數組中將拷貝所有就緒的事件,從內核事件表中。events的成員是struct epoll_event類型,一般包含events(其值是如:EPOLLIN、EPOLLOUT等)。還有一個是data.fd,包含拷貝的事件對應的socket,如果是伺服器監聽socket,說明是有用戶連接到這個socket,如果是其他已經連接好的socket,說明是有數據要發送或者接收。

如果事件數組中的數據得到處理,那麼內核事件表中的事件就會被刪除,下次wait就沒有那些socket的事件了。 

 

實例代碼:

epoll.c

#include <stdio.h>

#include <sys/epoll.h>

#include <sys/socket.h>

#include <stdlib.h>

#include <netinet/in.h>                 //包含sockaddr_in定義

#include <errno.h>

#include <string.h>                     //包含memset strncpy

 

 

int main(int argc,char* argv[])   //主函數

{

 

      int epfd1;int result;

      int server_len,client_len;

      int server_sockfd,client_sockfd;

      struct sockaddr_in server_address;       //定義在 <netinet/in.h>

      struct sockaddr_in client_address;

      struct epoll_event ev1;

      struct epoll_event ev[20];

      int epollreturn;

      int i,j,res;

      int sockfd;

      char ch = '0';

      char buff[1024];

     

      server_address.sin_family = AF_INET;

      server_address.sin_addr.s_addr = inet_addr("192.168.131.129");

      server_sockfd = socket(AF_INET,SOCK_STREAM,0);

      server_address.sin_port = htons(9734);

      server_len = sizeof(server_address);

      client_len = sizeof(client_address);

     

      result = bind(server_sockfd,(struct sockaddr*)&server_address,server_len);

      if(result!=0)

      {

           printf("bind failed\n");

           exit(1);                             //在stdlib.h

      }   

 

     

     

      epfd1 = epoll_create(10000);

      ev1.data.fd = server_sockfd;

      ev1.events = EPOLLIN;

      /*

      printf("%08x\n",EPOLLIN);

      printf("%08x\n",EPOLLOUT);

      printf("%08x\n",EPOLLPRI);

      printf("%08x\n",EPOLLERR);

      printf("%08x\n",EPOLLHUP);

      printf("%08x\n",EPOLLET);

      printf("%08x\n",EPOLLONESHOT);

      */

      epoll_ctl(epfd1,EPOLL_CTL_ADD,server_sockfd,&ev1);

     

     

     

      result = listen(server_sockfd,5);

      if(result!=0)

      {

           printf("listen failed\n");

           exit(1);

      }

     

      memset(buff,0,1024);

      strncpy(buff,"this is server",14);

     

      for(;;)

      {   

           epollreturn  = epoll_wait(epfd1,ev,20,4000);

           printf("epollreturn is %d\n",epollreturn);

           if(epollreturn>0)

           {

                 for(i=0;i<epollreturn;i++)

                 {

                     

                     if(ev[i].data.fd==server_sockfd)//如果新監測到一個SOCKET用戶連接到了綁定的SOCKET埠,建立新的連接。

                      {

                           

                            client_sockfd = accept(server_sockfd,(struct sockaddr *)&client_address, &client_len);//沒有計算client_len的值,會導致accept返回-1

                            printf("accept one client,socket:%d\n",client_sockfd);                        

                            ev1.data.fd=client_sockfd;

                            ev1.events=EPOLLIN;

                            epoll_ctl(epfd1,EPOLL_CTL_ADD,client_sockfd,&ev1);

                           

                            //ev1.data.fd=client_sockfd;

                            //ev1.events=EPOLLOUT;

                            //epoll_ctl(epfd1,EPOLL_CTL_ADD,client_sockfd,&ev1); //註冊

                      }

                      else if(ev[i].events&EPOLLIN)//如果是已經連接的用戶,收到數據,那麼進行讀入。

                      {

                            sockfd = ev[i].data.fd;

                            if (sockfd < 0)

                            {

                                  printf("EPOLLIN,sockfd < 0\n");

                                  continue;

                            }

                            res = recv(sockfd,&ch,1,0);

                                                       

                            if (res < 0)

                            {

                                 if (errno == ECONNRESET)

                                  {

                                       close(sockfd);

                                       ev[i].data.fd = -1;

                                       printf("EPOLLIN,res<0,errno == ECONNRESET\n");

                                  }

                                  else

                                  printf("EPOLLIN,recv error,res <0\n");

                            }

                            else if (res == 0)

                            {

                                  close(sockfd);                    //個測試發現關閉socket,epoll隊列中就不再監視這個socket了,似乎不需要刪除監視

                                  ev[i].data.fd = -1;

                                  printf("EPOLLIN,res == 0\n");

                                  ev1.data.fd=sockfd;

                                  ev1.events=EPOLLIN;

                                  epoll_ctl(epfd1,EPOLL_CTL_DEL,sockfd,&ev1);

                            }

                            else

                            {

                                  printf("EPOLLIN,receive one char %c,socket is %d\n",ch,sockfd);

                            }

 

                            ev1.data.fd=sockfd; 

                            ev1.events=EPOLLOUT;

                            epoll_ctl(epfd1,EPOLL_CTL_MOD,sockfd,&ev1);

                            /**/

                      }

                     

                      else if(ev[i].events&EPOLLOUT) // 監測數據發送的原理是,對端調用recv,通知到伺服器端,通知epoll,這個socket有數據要發。

                      {

                            sockfd = ev[i].data.fd;

                            res = send(sockfd,buff,102,0);

                            if(res==-1)

                            {

                                  printf("send error,res is %d\n",res);

                                  close(sockfd);

                                  ev1.data.fd=sockfd;

                                  ev1.events=EPOLLOUT;

                                  epoll_ctl(epfd1,EPOLL_CTL_DEL,sockfd,&ev1);

                            }

                           

                           

                            ev1.data.fd=sockfd; //設置用於讀操作的文件描述符

                            ev1.events=EPOLLIN; //設置用於註測的讀操作事件

                            epoll_ctl(epfd1,EPOLL_CTL_MOD,sockfd,&ev1);  //修改sockfd上要處理的事件為EPOLIN

                           

                      }

                

                

                 }

          

          

           }

          

     

     

      }

      return 0;

}

 

 

client2.c

#include <sys/types.h>

#include <sys/socket.h>

#include <stdio.h>

#include <netinet/in.h>

#include <arpa/inet.h>

#include <unistd.h>

#include <stdlib.h>

#include <string.h>                     //包含memset strncpy

 

int main(int argc,char* argv[])                         //這是網路套接字,比unix套接字的測試要簡單

{                                                       //測試方法,把client2和server2放在相同或者不同文件夾下都可以

      int sockfd;                                         // 一個終端運行./client2  另外一個終端運行  ./server2 就可以看到結果了

      int len,i,res;

      struct sockaddr_in address;

      int result;

      char ch = 'A';

      char buff[1024];

     

      sockfd = socket(AF_INET,SOCK_STREAM,0);              //奇怪,一個client運行多個版本,每次獲取的居然是同一個socket。改個名字也不行

      printf("socket is %d\n",sockfd);

      address.sin_family = AF_INET;

      address.sin_addr.s_addr = inet_addr("192.168.131.129");

      address.sin_port = htons(9734);

      len = sizeof(address);

     

      result = connect(sockfd,(struct sockaddr*)&address,len);

      if(result == -1)

      {

           perror("oops:client1");

           exit(-1);

      }

     

      memset(buff,0,1024);

      i = 0;

      //for(i=0;i<10;i++)

      for(;;)

      {  

           res = send(sockfd,&ch,1,0);

           if(res==-1)

           {

                 printf("send error,res is %d,exiting program\n",res);

                 close(sockfd);

                 return(-1);

           }

           /**/ 

           i++;

            memset(buff,0,102);

           res = recv(sockfd,buff,102,0); 

          

            //if((res==-1)||(res==0))

           if(res==-1)                       

           {

                 printf("recv error,res is %d,exiting program\n",res);

                 close(sockfd);

                 return(-1);

           }

           printf("socket:%d,buff is %s,i is %d\n",sockfd,buff,i);

           /**/

      }

      //scanf("%s",buff);

      printf("exiting program\n");

      close(sockfd);  

      return 0;

}

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 學習目標 分析Makefile文件,瞭解內核中的哪些文件被編譯,如何被編譯,連接時順序如何確定! Linux內核源碼中包含很多的Makefile文件,這些Makefile文件又包含其它的一些文件,比如配置信息、通用規則等等。我們可以把內核中的Makefile文件分為5類,如下表所示: 配置文件,在執 ...
  • TCP埠的連通性 方法一:TCP協議是面向連接的,可以直接通過telnet命令連接 方法二:使用nc(netcat)命令 nc相關option: UDP埠的連通性 UDP協議是無連接的,不需要握手建立連接,數據發送後,server端也不會返回確認信息。 方法一:使用nc(netcat)命令 總結 ...
  • 什麼是別名 在管理和維護Linux系統的過程中,將會使用到大量命令,有一些很長的命令或用法經常被用到,重覆而頻繁的輸入某個很長命令或用法是不可取的。這時可以使用 別名 功能將這個過程簡單化。 Linux系統下 有的命令 如 rm cp mv 等 這些命令是刪除 移動之類的 使用時要謹慎 也可以通過設 ...
  • 不知道作為運維的你有沒有體會過這樣一種情況: 當某天你的伺服器發生異常情況,例如某個文件莫名被刪除了,或者某個文件被人私自篡改,甚至是發生安全事件等等,這時你的經理找到你要你查個水落石出,於是你想看看history里有沒有一些異常的操作,當你在終端里敲完history命令之後,看到的結果,卻敵我難分 ...
  • 註意:雲虛擬主機和雲伺服器(ECS)不是同一個產品,請註意分別. 雲伺服器ECS: 雲虛擬主機: 我用的是雲虛擬主機也是第二個,版本是window server 聲明:預設,已經把功能變數名稱【已備案】綁定到自己的雲虛擬主機上了,【功能變數名稱綁定暫時不涉及】也就是說可以直接通過功能變數名稱訪問到你的阿裡雲虛擬主機 如下圖 ...
  • centos為例 一, 如下: 如果這個地方卡住了的話也許是你上次改了passwd文件,這個是其中一個情況。 重啟客戶端在引導的時候按e或者F5就行 摁e進去不要摁enter! 選擇k開頭的摁e 輸入init=/bin/sh 確定 摁b進入系統 進去這個頁面後 輸入mount -orw,unmoun ...
  • linux命令入門 忘記密碼啦!!! 忘記密碼教程!!! 一,入門命令講解:不詳解 教你們忘記密碼(我原來密碼就是123456,忘記是不可能的!假裝忘記的樣子 0.0) 現在我們忘記密碼了!對忘記密碼了。我忘記密碼了 重啟系統! 對哦我忘記密碼了 那你關機在啟動啊啊! 在這裡的時候就按e鍵或者f5 ...
  • 工作原因開始使用Ubuntu.桌面環境為GNOME,不過亮度調整和桌面環境沒多大關係. 思路: 不管是GNOME還是Unity,都會嘗試自己去接管亮度調整,也就是去 /sys/class/backlight下麵去找brightness.所以說亮度無法調整大概可以歸類到兩種情況: 1.sys/clas ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...