成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

TCP/IP網(wǎng)絡(luò)編程 --優(yōu)于select的epoll

網(wǎng)絡(luò) 網(wǎng)絡(luò)管理
關(guān)于并發(fā)服務(wù)器中的I/O復(fù)用實現(xiàn)方式,前面我們講過select的方式,但select的性能比較低,并不適合以Web服務(wù)器端開發(fā)為主流的現(xiàn)代開發(fā)環(huán)境。因此就有了Linux下的epoll,BSD的kqueue,Solaris的/dev/poll和Windows的IOCP等復(fù)用技術(shù)。本章就來講講Linux下的epoll技術(shù)。

關(guān)于并發(fā)服務(wù)器中的I/O復(fù)用實現(xiàn)方式,前面我們講過select的方式,但select的性能比較低,并不適合以Web服務(wù)器端開發(fā)為主流的現(xiàn)代開發(fā)環(huán)境。因此就有了Linux下的epoll,BSD的kqueue,Solaris的/dev/poll和Windows的IOCP等復(fù)用技術(shù)。本章就來講講Linux下的epoll技術(shù)。

epoll理解及應(yīng)用

基于select的I/O復(fù)用技術(shù)速度慢的原因:

1,調(diào)用select函數(shù)后常見的針對所有文件描述符的循環(huán)語句。它每次事件發(fā)生需要遍歷所有文件描述符,找出發(fā)生變化的文件描述符。(以前寫的示例沒加循環(huán))

2,每次調(diào)用select函數(shù)時都需要向該函數(shù)傳遞監(jiān)視對象信息。即每次調(diào)用select函數(shù)時向操作系統(tǒng)傳遞監(jiān)視對象信息,至于為什么要傳?是因為我們監(jiān)視的套接字變化的函數(shù),而套接字是操作系統(tǒng)管理的。(這個才是最耗效率的)

注釋:基于這樣的原因并不是說select就沒用了,在這樣的情況下就適合選用select:1,服務(wù)端接入者少 2,程序應(yīng)具有兼容性。

epoll是怎么優(yōu)化select問題的:

1,每次發(fā)生事件它不需要循環(huán)遍歷所有文件描述符,它把發(fā)生變化的文件描述符單獨集中到了一起。

2,僅向操作系統(tǒng)傳遞1次監(jiān)視對象信息,監(jiān)視范圍或內(nèi)容發(fā)生變化時只通知發(fā)生變化的事項。

實現(xiàn)epoll時必要的函數(shù)和結(jié)構(gòu)體

  1. 函數(shù): 
  2.  
  3. epoll_create:創(chuàng)建保存epoll文件描述符的空間,該函數(shù)也會返回文件描述符,所以終止時,也要調(diào)用close函數(shù)。(創(chuàng)建內(nèi)存空間) 
  4.  
  5. epoll_ctl:向空間注冊,添加或修改文件描述符。(注冊監(jiān)聽事件) 
  6.  
  7. epoll_wait:與select函數(shù)類似,等待文件描述符發(fā)生變化。(監(jiān)聽事件回調(diào)) 
  8.  
  9. 結(jié)構(gòu)體: 
  10.  
  11. struct epoll_event  
  12.   
  13. {  
  14.   
  15. __uint32_t events;  
  16.   
  17. epoll_data_t data;  
  18.   
  19. }  
  20.   
  21. typedef union epoll_data  
  22.   
  23. {  
  24.   
  25. void *ptr;  
  26.   
  27. int fd;  
  28.   
  29. __uinit32_t u32;  
  30.   
  31. __uint64_t u64;  
  32.   
  33. } epoll_data_t;  

基于epoll的回聲服務(wù)器端

  1. // 
  2.  
  3. // main.cpp 
  4.  
  5. // hello_server 
  6.  
  7. // 
  8.  
  9. // Created by app05 on 15-10-19. 
  10.  
  11. // Copyright (c) 2015年 app05. All rights reserved. 
  12.  
  13. // 
  14.  
  15. #include 
  16.  
  17. #include 
  18.  
  19. #include 
  20.  
  21. #include 
  22.  
  23. #include 
  24.  
  25. #include 
  26.  
  27. #include 
  28.  
  29. #define BUF_SIZE 100 
  30.  
  31. #define EPOLL_SIZE 50 
  32.  
  33. void error_handling(char *buf); 
  34.  
  35. int main(int argc, const char * argv[]) { 
  36.  
  37. int serv_sock, clnt_sock; 
  38.  
  39. struct sockaddr_in serv_adr, clnt_adr; 
  40.  
  41. socklen_t adr_sz; 
  42.  
  43. int str_len, i; 
  44.  
  45. char buf[BUF_SIZE]; 
  46.  
  47. //類似select的fd_set變量查看監(jiān)視對象的狀態(tài)變化,epoll_event結(jié)構(gòu)體將發(fā)生變化的文件描述符單獨集中到一起 
  48.  
  49. struct epoll_event *ep_events; 
  50.  
  51. struct epoll_event event; 
  52.  
  53. int epfd, event_cnt; 
  54.  
  55. if(argc != 2) 
  56.  
  57.  
  58. printf("Usage: %s \n", argv[0]); 
  59.  
  60. exit(1); 
  61.  
  62.  
  63. serv_sock = socket(PF_INET, SOCK_STREAM, 0); 
  64.  
  65. if(serv_sock == -1) 
  66.  
  67. error_handling("socket() error"); 
  68.  
  69. memset(&serv_adr, 0, sizeof(serv_adr)); 
  70.  
  71. serv_adr.sin_family = AF_INET; 
  72.  
  73. serv_adr.sin_addr.s_addr = htonl(INADDR_ANY); 
  74.  
  75. serv_adr.sin_port = htons(atoi(argv[1])); 
  76.  
  77. if(bind(serv_sock, (struct sockaddr *) &serv_adr, sizeof(serv_adr)) == -1) 
  78.  
  79. error_handling("bind() error"); 
  80.  
  81. if(listen(serv_sock, 5) == -1) 
  82.  
  83. error_handling("listen() error"); 
  84.  
  85. //創(chuàng)建文件描述符的保存空間稱為“epoll例程” 
  86.  
  87. epfd = epoll_create(EPOLL_SIZE); 
  88.  
  89. ep_events = malloc(sizeof(struct epoll_event) *EPOLL_SIZE); 
  90.  
  91. //添加讀取事件的監(jiān)視(注冊事件) 
  92.  
  93. event.events = EPOLLIN; //讀取數(shù)據(jù)事件 
  94.  
  95. event.data.fd = serv_sock; 
  96.  
  97. epoll_ctl(epdf, EPOLL_CTL_ADD, serv_sock, &event); 
  98.  
  99. while (1) 
  100.  
  101.  
  102. //響應(yīng)事件,返回發(fā)生事件的文件描述符數(shù) 
  103.  
  104. event_cnt = epoll_wait(epfd, ep_events, EPOLL_SIZE, -1); //傳-1時,一直等待直到事件發(fā)生 
  105.  
  106. if(event_cnt == -1) 
  107.  
  108.  
  109. puts("epoll_wait() error"); 
  110.  
  111. break
  112.  
  113.  
  114. //服務(wù)端套接字和客服端套接字 
  115.  
  116. for (i = 0; i < event_cnt; i++) { 
  117.  
  118. if(ep_events[i].data.fd == serv_sock)//服務(wù)端與客服端建立連接 
  119.  
  120.  
  121. adr_sz = sizeof(clnt_adr); 
  122.  
  123. clnt_sock = accept(serv_sock, (struct sockaddr *)&clnt_adr, &adr_sz); 
  124.  
  125. event.events = EPOLLIN; 
  126.  
  127. event.data.fd = clnt_sock; 
  128.  
  129. epoll_ctl(epfd, EPOLL_CTL_ADD, clnt_sock, &event); 
  130.  
  131. printf("connected client: %d \n", clnt_sock); 
  132.  
  133.  
  134. else //連接之后傳遞數(shù)據(jù) 
  135.  
  136.  
  137. str_len = read(ep_events[i].data.fd, buf, BUF_SIZE); 
  138.  
  139. if(str_len == 0) 
  140.  
  141.  
  142. //刪除事件 
  143.  
  144. epoll_ctl(epfd, EPOLL_CTL_DEL, ep_events[i].data.fd, NULL); 
  145.  
  146. close(ep_events[i].data.fd); 
  147.  
  148. printf("closed client: %d \n", ep_events[i].data.fd); 
  149.  
  150.  
  151. else 
  152.  
  153.  
  154. write(ep_events[i].data.fd, buf, str_len); 
  155.  
  156.  
  157.  
  158.  
  159.  
  160. close(serv_sock); 
  161.  
  162. close(epfd); 
  163.  
  164. return 0; 
  165.  
  166.  
  167. void error_handling(char *message) 
  168.  
  169.  
  170. fputs(message, stderr); 
  171.  
  172. fputc('\n', stderr); 
  173.  
  174. exit(1); 
  175.  
  176. }

條件觸發(fā)和邊緣觸發(fā)

什么是條件觸發(fā)和邊緣觸發(fā)?它們是指事件響應(yīng)的方式,epoll默認是條件觸發(fā)的方式。條件觸發(fā)是指:只要輸入緩沖中有數(shù)據(jù)就會一直通知該事件,循環(huán)響應(yīng)epoll_wait。而邊緣觸發(fā)是指:輸入緩沖收到數(shù)據(jù)時僅注冊1次該事件,即使輸入緩沖中還留有數(shù)據(jù),也不會再進行注冊,只響應(yīng)一次。

邊緣觸發(fā)相對條件觸發(fā)的優(yōu)點:可以分離接收數(shù)據(jù)和處理數(shù)據(jù)的時間點,從實現(xiàn)模型的角度看,邊緣觸發(fā)更有可能帶來高性能。

將上面epoll實例改為邊緣觸發(fā):

1,首先改寫 event.events = EPOLLIN | EPOLLET; (EPOLLIN:讀取數(shù)據(jù)事件 EPOLLET:邊緣觸發(fā)方式)

2,邊緣觸發(fā)只響應(yīng)一次接收數(shù)據(jù)事件,所以要一次性全部讀取輸入緩沖中的數(shù)據(jù),那么就需要判斷什么時候數(shù)據(jù)讀取完了?Linux聲明了一個全局的變量:int errno; (error.h中),它能記錄發(fā)生錯誤時提供額外的信息。這里就可以用它來判斷是否讀取完數(shù)據(jù):

  1. str_len = read(...); 
  2.  
  3. if(str_len < 0) 
  4.  
  5.  
  6. if(errno == EAGAIN) //讀取輸入緩沖中的全部數(shù)據(jù)的標(biāo)志 
  7.  
  8. break
  9.  
  10. }

3,邊緣觸發(fā)方式下,以阻塞方式工作的read&write有可能會引起服務(wù)端的長時間停頓。所以邊緣觸發(fā)一定要采用非阻塞的套接字數(shù)據(jù)傳輸形式。那么怎么將套接字的read,write數(shù)據(jù)傳輸形式修改為非阻塞模式呢?

//fd套接字文件描述符,將此套接字數(shù)據(jù)傳輸模式修改為非阻塞

  1. void setnonblockingmode(int fd) 
  2.  
  3.  
  4. int flag = fcntl(fd, F_GETFL,0); //得到套接字原來屬性 
  5.  
  6. fcntl(fd, F_SETFL, flag | O_NONBLOCK);//在原有屬性基礎(chǔ)上設(shè)置添加非阻塞模式 
  7.  
責(zé)任編輯:何妍 來源: CSDN博客
相關(guān)推薦

2019-09-18 20:07:06

AndroidTCP協(xié)議

2015-04-24 09:48:59

TCPsocketsocket編程

2015-10-19 09:34:42

TCPIP網(wǎng)絡(luò)協(xié)議

2019-11-08 14:47:49

TCPIP網(wǎng)絡(luò)

2015-10-16 09:33:26

TCPIP網(wǎng)絡(luò)協(xié)議

2019-04-08 08:44:10

TCPIP網(wǎng)絡(luò)協(xié)議

2009-04-09 10:11:00

TCPIP通訊

2010-09-09 16:28:19

2015-10-27 09:40:31

TCPIP網(wǎng)絡(luò)協(xié)議

2025-02-06 09:47:33

2010-09-09 16:21:32

TCP IP網(wǎng)絡(luò)協(xié)議

2012-09-24 15:13:50

C#網(wǎng)絡(luò)協(xié)議TCP

2023-06-01 07:55:56

TCP/IP網(wǎng)絡(luò)模型

2013-10-28 09:24:34

SDN軟件定義網(wǎng)絡(luò)TCP

2012-12-03 11:50:44

TCPIP網(wǎng)絡(luò)流量

2019-07-16 11:06:09

TCP四次揮手半關(guān)閉

2021-03-17 09:51:31

網(wǎng)絡(luò)編程TCP網(wǎng)絡(luò)協(xié)議

2019-04-29 07:53:11

TCP數(shù)據(jù)包TCP網(wǎng)絡(luò)編程

2022-02-22 08:55:29

SelectPoll/ Epoll

2022-10-08 00:00:00

DNS地址網(wǎng)關(guān)
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 日韩中文一区 | 偷拍第一页 | 国产清纯白嫩初高生在线播放视频 | 久久久久久久久久久福利观看 | 亚洲成人久久久 | 久久国产欧美日韩精品 | www国产亚洲精品久久网站 | 午夜免费福利电影 | 范冰冰一级做a爰片久久毛片 | 91玖玖 | 国产高清一区二区 | 亚洲综合色自拍一区 | 91视视频在线观看入口直接观看 | 亚洲天堂精品久久 | 国产一区二区在线视频 | 国产特黄一级 | 色婷婷综合网 | 国产精品久久久久久久久久了 | 中文字幕成人av | 少妇一区二区三区 | 在线男人天堂 | av网站免费 | 中文字幕高清 | 精品久久久久久红码专区 | 国际精品鲁一鲁一区二区小说 | 久久久久久免费免费 | 国产视频一区二区 | 国产黄色大片 | 好好的日在线视频 | 91精品国产91 | 久久久精品一区二区三区 | 亚洲一区二区三区在线视频 | 久久五月婷 | 羞羞网站免费 | 欧美日韩免费在线 | 国产一区二区小视频 | 欧美久久精品一级黑人c片 91免费在线视频 | 国产在线97| 在线视频一区二区三区 | 成人在线精品视频 | 久久se精品一区精品二区 |