• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            doing5552

            記錄每日點(diǎn)滴,不枉人生一世

              C++博客 :: 首頁 :: 聯(lián)系 :: 聚合  :: 管理
              73 Posts :: 0 Stories :: 94 Comments :: 0 Trackbacks

            公告

            常用鏈接

            留言簿(24)

            我參與的團(tuán)隊(duì)

            最新隨筆

            搜索

            •  

            積分與排名

            • 積分 - 455263
            • 排名 - 48

            最新隨筆

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            用epoll實(shí)現(xiàn)異步的Echo服務(wù)器
            2010年05月07日 星期五 下午 06:48
            epoll是Kernel 2.6后新加入的事件機(jī)制,在高并發(fā)條件下,遠(yuǎn)優(yōu)于select.

            用個(gè)硬件中的例子吧,可能不太恰當(dāng):epoll相當(dāng)于I/O中斷(有的時(shí)候才相應(yīng)),而select相當(dāng)于輪詢(總要反復(fù)查詢)。

            其實(shí)epoll比slect好用很多,主要一下幾個(gè)用法。

            struct epoll_event ; epoll事件體,事件發(fā)生時(shí)候你可以得到一個(gè)它。其中epoll_event.data.fd可以存儲(chǔ)關(guān)聯(lián)的句柄,epoll_event.event 是監(jiān)聽標(biāo)志,常用的有EPOLLIN (有數(shù)據(jù),可以讀)、EPOLLOUT(有數(shù)據(jù),可以寫)EPOLLET(有事件,通用);

            (1)創(chuàng)建epoll句柄

            int epFd = epoll_create(EPOLL_SIZE);

            (2)加入一個(gè)句柄到epoll的監(jiān)聽隊(duì)列

            ev.data.fd = serverFd;
            ev.events = EPOLLIN | EPOLLET;
            epoll_ctl(epFd, EPOLL_CTL_ADD, serverFd, &ev);

            上面的fd是你要綁定給事件發(fā)生時(shí)候使用的fd,到時(shí)候只能操作這個(gè),下面是事件類型。

            使用epoll_ctl添加到之中,EPOLL_CTL_ADD是epoll控制類型,這里監(jiān)聽的fd和給event的fd一般相同。

            (3)等待event返回

            int nfds = epoll_wait(epFd, evs, EVENT_ARR, -1);

            傳入的evs是epoll_event的數(shù)組,EVENT_ARR應(yīng)當(dāng)是不超過這個(gè)數(shù)組的長(zhǎng)度。返回nfds的是不超過EVENT_ARR的數(shù)值,表示本次等待到了幾個(gè)事件。

            (4)遍歷事件

            注意,這里遍歷的事件是肯定已經(jīng)發(fā)生了的,而select中遍歷的是每個(gè)fd,而fd不一定在FDSET中(即不一定有讀事件發(fā)生)!這是效率最大的差別所在!

            for (int i = 0; i < nfds; i++)

            {

            //do something

            }

            (5)其他技巧

            對(duì)事件是否是serverFd判斷,如果是,進(jìn)行accept并加入epoll監(jiān)聽隊(duì)列,要設(shè)置異步讀取。

            如果evts[i]&EPOLLIN,表示可讀,使用read進(jìn)行試探,如果>0表示連接沒有關(guān)閉,否則連接已經(jīng)關(guān)閉(出發(fā)事件又讀取不到東西,表示socket關(guān)閉!)。如果<0出錯(cuò)。如果>0,需要繼續(xù)讀取直到為0,但是注意這里的為0是在第一次read不為0的前提下,畢竟我們?cè)O(shè)置了異步讀取,暫時(shí)沒有數(shù)據(jù)可以讀就返回0了!而如果第一次返回0,那么就是關(guān)閉吧!

            注意關(guān)閉要移出出epoll并且close(clientFd)

            羅嗦了好多,看代碼!
            查看源代碼
            打印幫助
            001    /*
            002     * main.cc
            003     *
            004     *  Created on: 2009-11-30
            005     *      Author: liheyuan
            006     *    Describe: epoll實(shí)現(xiàn)阻塞模式服務(wù)器(Echo服務(wù)器)
            007     *
            008     *   Last Date: 2009-11-30
            009     *   CopyRight: 2009 @ ICT LiHeyuan
            010     */
            011    #include <stdio.h>
            012    #include <stdlib.h>
            013    #include <unistd.h>
            014    #include <fcntl.h>
            015    #include <arpa/inet.h>
            016    #include <netinet/in.h>
            017    #include <sys/epoll.h>
            018    #include <errno.h>
            019    
            020    #define EPOLL_SIZE 10
            021    #define EVENT_ARR 20
            022    #define BACK_QUEUE 10
            023    #define PORT 18001
            024    #define BUF_SIZE 16
            025    
            026    void setnonblocking(int sockFd) {
            027        int opt;
            028    
            029        //獲取sock原來的flag
            030        opt = fcntl(sockFd, F_GETFL);
            031        if (opt < 0) {
            032            printf("fcntl(F_GETFL) fail.");
            033            exit(-1);
            034        }
            035    
            036        //設(shè)置新的flag,非阻塞
            037        opt |= O_NONBLOCK;
            038        if (fcntl(sockFd, F_SETFL, opt) < 0) {
            039            printf("fcntl(F_SETFL) fail.");
            040            exit(-1);
            041        }
            042    }
            043    
            044    int main() {
            045    
            046        int serverFd;
            047    
            048        //創(chuàng)建服務(wù)器fd
            049        serverFd = socket(AF_INET, SOCK_STREAM, 0);
            050        setnonblocking(serverFd);
            051    
            052        //創(chuàng)建epoll,并把 serverFd放入監(jiān)聽隊(duì)列
            053        int epFd = epoll_create(EPOLL_SIZE);
            054        struct epoll_event ev, evs[EVENT_ARR];
            055        ev.data.fd = serverFd;
            056        ev.events = EPOLLIN | EPOLLET;
            057        epoll_ctl(epFd, EPOLL_CTL_ADD, serverFd, &ev);
            058    
            059        //綁定服務(wù)器端口
            060        struct sockaddr_in serverAddr;
            061        socklen_t serverLen = sizeof(struct sockaddr_in);
            062        serverAddr.sin_addr.s_addr = htonl(INADDR_ANY);
            063        serverAddr.sin_port = htons(PORT);
            064        if (bind(serverFd, (struct sockaddr *) &serverAddr, serverLen)) {
            065            printf("bind() fail.\n");
            066            exit(-1);
            067        }
            068    
            069        //打開監(jiān)聽
            070        if (listen(serverFd, BACK_QUEUE)) {
            071            printf("Listen fail.\n");
            072            exit(-1);
            073        }
            074    
            075        //死循環(huán)處理
            076        int clientFd;
            077        sockaddr_in clientAddr;
            078        socklen_t clientLen;
            079        char buf[BUF_SIZE];
            080        while (1) {
            081            //等待epoll事件的到來,最多取EVENT_ARR個(gè)事件
            082            int nfds = epoll_wait(epFd, evs, EVENT_ARR, -1);
            083            //處理事件
            084            for (int i = 0; i < nfds; i++) {
            085                if (evs[i].data.fd == serverFd && evs[i].data.fd & EPOLLIN) {
            086                    //如果是serverFd,表明有新連接連入
            087                    if ((clientFd = accept(serverFd,
            088                            (struct sockaddr *) &clientAddr, &clientLen)) < 0) {
            089                        printf("accept fail.\n");
            090                    }
            091                    printf("Connect from %s:%d\n", inet_ntoa(clientAddr.sin_addr),
            092                            htons(clientAddr.sin_port));
            093                    setnonblocking(clientFd);
            094                    //注冊(cè)accept()到的連接
            095                    ev.data.fd = clientFd;
            096                    ev.events = EPOLLIN | EPOLLET;
            097                    epoll_ctl(epFd, EPOLL_CTL_ADD, clientFd, &ev);
            098                } else if (evs[i].events & EPOLLIN) {
            099                    //如果不是serverFd,則是client的可讀
            100                    if ((clientFd = evs[i].data.fd) > 0) {
            101                        //先進(jìn)行試探性讀取
            102                        int len = read(clientFd, buf, BUF_SIZE);
            103                        if (len > 0) {
            104                            //有數(shù)據(jù)可以讀,Echo寫入
            105                            do {
            106                                if (write(clientFd, buf, len) < 0) {
            107                                    printf("write() fail.\n");
            108                                }
            109                                len = read(clientFd, buf, BUF_SIZE);
            110                            } while (len > 0);
            111                        } else if (len == 0) {
            112                            //出發(fā)了EPOLLIN事件,卻沒有可以讀取的,表示斷線
            113                            printf("Client closed at %d\n", clientFd);
            114                            epoll_ctl(epFd, EPOLL_CTL_DEL, clientFd, &ev);
            115                            close(clientFd);
            116                            evs[i].data.fd = -1;
            117                            break;
            118                        } else if (len == EAGAIN) {
            119                            continue;
            120                        } else {
            121                            //client讀取出錯(cuò)
            122                            printf("read() fail.");
            123                        }
            124                    }
            125                } else {
            126                    printf("other event.\n");
            127                }
            128            }
            129        }
            130    
            131        return 0;
            132    }
            posted on 2010-07-18 21:14 doing5552 閱讀(807) 評(píng)論(0)  編輯 收藏 引用

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            大伊人青草狠狠久久| 色狠狠久久综合网| 99久久精品国产一区二区蜜芽| 成人国内精品久久久久影院| 国产精品成人久久久久三级午夜电影 | 亚洲AV日韩精品久久久久久| 久久精品中文字幕无码绿巨人| 国产精品久久影院| 日本精品久久久久影院日本 | 久久久一本精品99久久精品88| 91久久精一区二区三区大全| 色婷婷噜噜久久国产精品12p| 丁香色欲久久久久久综合网| 国产精品狼人久久久久影院| 久久久久av无码免费网| 精品久久久久久无码中文字幕| 久久无码AV一区二区三区| 99久久国产主播综合精品| 亚洲精品高清国产一线久久| 久久免费视频一区| 国产精品久久久99| 国产精品岛国久久久久| 99蜜桃臀久久久欧美精品网站| 亚洲天堂久久精品| 久久精品国产亚洲AV高清热| 久久久久久国产精品无码下载 | 精品久久人人妻人人做精品| 欧洲成人午夜精品无码区久久| 日本精品久久久久久久久免费| 成人亚洲欧美久久久久| 久久99国产精品久久久| 久久久久女人精品毛片| 久久青青草原精品国产| 久久久久国产精品熟女影院| 久久SE精品一区二区| 久久99热这里只有精品国产| 久久亚洲精品成人无码网站| 香港aa三级久久三级老师2021国产三级精品三级在| 久久九九青青国产精品| 亚洲欧美精品伊人久久| 久久AⅤ人妻少妇嫩草影院|