• <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>

            天下

            記錄修行的印記

            epoll中的LT和ET

            LT讓我想起了linux內核的中斷處理,設置中斷觸發標志


            電平觸發,就是只有高電平(或者低電平)的時候才做指定的事,  
            邊沿觸發,就是有高電平向低電平轉換,或者翻過來轉換,這個轉換過程觸發一個動作。

            電平就是邏輯上的0,1觸發,  
            而邊沿就是脈沖突變觸發,邏輯上就是0-1或是1-
            0  也就是上樓的那位表示的  
              
            通俗點講吧  
            電平就是電壓,高電平就是高電壓,低電平就是低電壓
            高電平觸發就是當電壓為高時觸發
            邊沿觸發就是當電壓由高變低或由低變高時觸發
            上升沿觸發  就是當電壓從低變高時觸發  
            下降沿觸發  就是當電壓從高變低時觸發
                
                
            轉個epoll的例子
            http:
            //www.shnenglu.com/API/archive/2013/12/01/204535.html    

            #include 
            <iostream>
            #include 
            <sys/socket.h>
            #include 
            <sys/epoll.h>
            #include 
            <netinet/in.h>
            #include 
            <arpa/inet.h>
            #include 
            <fcntl.h>
            #include 
            <unistd.h>
            #include 
            <stdio.h>
            #include 
            <pthread.h>

            #include 
            <errno.h>

            #define MAXLINE 10
            #define OPEN_MAX 100
            #define LISTENQ 20
            #define SERV_PORT 8006
            #define INFTIM 1000

            //線程池任務隊列結構體

            struct task{
              
            int fd; //需要讀寫的文件描述符

              
            struct task *next; //下一個任務

            };

            //用于讀寫兩個的兩個方面傳遞參數

            struct user_data{
              
            int fd;
              unsigned 
            int n_size;
              
            char line[MAXLINE];
            };

            //線程的任務函數

            void * readtask(void *args);
            void * writetask(void *args);


            //聲明epoll_event結構體的變量,ev用于注冊事件,數組用于回傳要處理的事件

            struct epoll_event ev,events[20];
            int epfd;
            pthread_mutex_t mutex;
            pthread_cond_t cond1;
            struct task *readhead=NULL,*readtail=NULL,*writehead=NULL;

            void setnonblocking(int sock)
            {
                 
            int opts;
                 opts
            =fcntl(sock,F_GETFL);
                 
            if(opts<0)
                 {
                      perror(
            "fcntl(sock,GETFL)");
                      exit(
            1);
                 }
                opts 
            = opts|O_NONBLOCK;
                 
            if(fcntl(sock,F_SETFL,opts)<0)
                 {
                      perror(
            "fcntl(sock,SETFL,opts)");
                      exit(
            1);
                 }
            }

            int main()
            {
                 
            int i, maxi, listenfd, connfd, sockfd,nfds;
                 pthread_t tid1,tid2;

                 
            struct task *new_task=NULL;
                 
            struct user_data *rdata=NULL;
                 socklen_t clilen;

                 pthread_mutex_init(
            &mutex,NULL);
                 pthread_cond_init(
            &cond1,NULL);
                 
            //初始化用于讀線程池的線程

                 pthread_create(
            &tid1,NULL,readtask,NULL);
                 pthread_create(
            &tid2,NULL,readtask,NULL);

                 
            //生成用于處理accept的epoll專用的文件描述符

                 epfd
            =epoll_create(256);

                 
            struct sockaddr_in clientaddr;
                 
            struct sockaddr_in serveraddr;
                 listenfd 
            = socket(AF_INET, SOCK_STREAM, 0);
                 
            //把socket設置為非阻塞方式

                 setnonblocking(listenfd);
                 
            //設置與要處理的事件相關的文件描述符

                 ev.data.fd
            =listenfd;
                 
            //設置要處理的事件類型

                 ev.events
            =EPOLLIN|EPOLLET;
                 
            //注冊epoll事件

                 epoll_ctl(epfd,EPOLL_CTL_ADD,listenfd,
            &ev);

                 bzero(
            &serveraddr, sizeof(serveraddr));
                 serveraddr.sin_family 
            = AF_INET;
                 serveraddr.sin_port
            =htons(SERV_PORT);
                 serveraddr.sin_addr.s_addr 
            = INADDR_ANY;
                 bind(listenfd,(sockaddr 
            *)&serveraddr, sizeof(serveraddr));
                 listen(listenfd, LISTENQ);

                 maxi 
            = 0;
                 
            for ( ; ; ) {
                      
            //等待epoll事件的發生

                      nfds
            =epoll_wait(epfd,events,20,500);
                      
            //處理所發生的所有事件

                    
            for(i=0;i<nfds;++i)
                    {
                           
            if(events[i].data.fd==listenfd)
                           {

                                connfd 
            = accept(listenfd,(sockaddr *)&clientaddr, &clilen);
                                
            if(connfd<0){
                                  perror(
            "connfd<0");
                                  exit(
            1);
                               }
                                setnonblocking(connfd);

                                
            char *str = inet_ntoa(clientaddr.sin_addr);
                                
            //std::cout<<"connec_ from >>"<<str<<std::endl;

                                
            //設置用于讀操作的文件描述符

                                ev.data.fd
            =connfd;
                                
            //設置用于注測的讀操作事件

                             ev.events
            =EPOLLIN|EPOLLET;
                                
            //注冊ev

                             epoll_ctl(epfd,EPOLL_CTL_ADD,connfd,
            &ev);
                           }
                        
            else if(events[i].events&EPOLLIN)
                        {
                                
            //printf("reading!/n");

                                
            if ( (sockfd = events[i].data.fd) < 0continue;
                                new_task
            =new task();
                                new_task
            ->fd=sockfd;
                                new_task
            ->next=NULL;
                                
            //添加新的讀任務

                                pthread_mutex_lock(
            &mutex);
                                
            if(readhead==NULL)
                                {
                                  readhead
            =new_task;
                                  readtail
            =new_task;
                                }
                                
            else
                                {
                                 readtail
            ->next=new_task;
                                  readtail
            =new_task;
                                }
                               
            //喚醒所有等待cond1條件的線程

                                pthread_cond_broadcast(
            &cond1);
                                pthread_mutex_unlock(
            &mutex);
                          }
                           
            else if(events[i].events&EPOLLOUT)
                           {
                             
            /*
                          rdata=(struct user_data *)events[i].data.ptr;
                             sockfd = rdata->fd;
                             write(sockfd, rdata->line, rdata->n_size);
                             delete rdata;
                             //設置用于讀操作的文件描述符
                             ev.data.fd=sockfd;
                             //設置用于注測的讀操作事件
                           ev.events=EPOLLIN|EPOLLET;
                             //修改sockfd上要處理的事件為EPOLIN
                           epoll_ctl(epfd,EPOLL_CTL_MOD,sockfd,&ev);
                         
            */
                           }

                      }

                 }
            }

            static int count111 = 0;
            static time_t oldtime = 0, nowtime = 0;
            void * readtask(void *args)
            {

               
            int fd=-1;
               unsigned 
            int n;
               
            //用于把讀出來的數據傳遞出去

               
            struct user_data *data = NULL;
               
            while(1){

                    pthread_mutex_lock(
            &mutex);
                    
            //等待到任務隊列不為空

                    
            while(readhead==NULL)
                         pthread_cond_wait(
            &cond1,&mutex);

                    fd
            =readhead->fd;
                    
            //從任務隊列取出一個讀任務

                    
            struct task *tmp=readhead;
                    readhead 
            = readhead->next;
                    delete tmp;
                    pthread_mutex_unlock(
            &mutex);
                    data 
            = new user_data();
                    data
            ->fd=fd;


                    
            char recvBuf[1024= {0};
                    
            int ret = 999;
                    
            int rs = 1;

                    
            while(rs)
                    {
                        ret 
            = recv(fd,recvBuf,1024,0);// 接受客戶端消息

                        
            if(ret < 0)
                        {
                            
            //由于是非阻塞的模式,所以當errno為EAGAIN時,表示當前緩沖區已無數據可//讀在這里就當作是該次事件已處理過。

                            
            if(errno == EAGAIN)
                            {
                                printf(
            "EAGAIN\n");
                                
            break;
                            }
                            
            else{
                                printf(
            "recv error!\n");

                                close(fd);
                                
            break;
                            }
                        }
                        
            else if(ret == 0)
                        {
                            
            // 這里表示對端的socket已正常關閉.

                            rs 
            = 0;
                        }
                        
            if(ret == sizeof(recvBuf))
                            rs 
            = 1// 需要再次讀取

                        
            else
                            rs 
            = 0;
                    }
                    
            if(ret>0){

                    
            //-------------------------------------------------------------------------------


                        data
            ->n_size=n;


                        count111 
            ++;

                        
            struct tm *today;
                        time_t ltime;
                        time( 
            &nowtime );

                        
            if(nowtime != oldtime){
                            printf(
            "%d\n", count111);
                            oldtime 
            = nowtime;
                            count111 
            = 0;
                        }

                        
            char buf[1000= {0};
                        sprintf(buf,
            "HTTP/1.0 200 OK\r\nContent-type: text/plain\r\n\r\n%s","Hello world!\n");
                        send(fd,buf,strlen(buf),
            0);
                        close(fd);


                   }
               }
            }

            posted on 2014-03-21 17:24 天下 閱讀(582) 評論(0)  編輯 收藏 引用 所屬分類: Linux編程

            <2025年5月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            導航

            統計

            常用鏈接

            留言簿(4)

            隨筆分類(378)

            隨筆檔案(329)

            鏈接

            最新隨筆

            搜索

            最新評論

            久久精品国产99国产精偷| 久久综合色老色| 久久天堂AV综合合色蜜桃网| 青青草原综合久久大伊人精品| 伊人久久成人成综合网222| 热re99久久6国产精品免费| 色婷婷久久综合中文久久蜜桃av | 狠狠色丁香久久综合婷婷| 久久综合狠狠综合久久激情 | 亚洲va中文字幕无码久久不卡| 久久人人爽人人澡人人高潮AV| 国产精品九九久久免费视频| 国产一区二区三区久久| 色婷婷久久综合中文久久一本| 国产激情久久久久影院| 久久精品国产亚洲AV香蕉| 亚洲国产成人久久一区WWW| 日日噜噜夜夜狠狠久久丁香五月| 中文字幕无码av激情不卡久久| 久久久黄色大片| 久久精品?ⅴ无码中文字幕| 久久国产成人精品麻豆 | 中文字幕久久波多野结衣av| 亚洲国产一成久久精品国产成人综合 | 国产精品日韩深夜福利久久| 色综合久久久久网| 99久久国产综合精品成人影院| 国产精品免费久久| 伊人久久亚洲综合影院| 日韩精品久久无码中文字幕| 久久精品国产亚洲AV香蕉| 狠狠色丁香久久婷婷综合蜜芽五月| 97久久国产露脸精品国产| 久久综合成人网| 亚洲精品美女久久久久99小说 | 久久香蕉超碰97国产精品| 亚洲精品成人网久久久久久| 国产精品久久国产精麻豆99网站 | 久久人人爽人人爽人人片AV麻豆| 大香网伊人久久综合网2020| 99久久久精品免费观看国产|