• <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:
            //m.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編程

            <2011年9月>
            28293031123
            45678910
            11121314151617
            18192021222324
            2526272829301
            2345678

            導航

            統計

            常用鏈接

            留言簿(4)

            隨筆分類(378)

            隨筆檔案(329)

            鏈接

            最新隨筆

            搜索

            最新評論

            av国内精品久久久久影院| 一本色综合久久| 久久久老熟女一区二区三区| 狠色狠色狠狠色综合久久| 久久综合久久伊人| 国产成人无码精品久久久免费 | 超级碰碰碰碰97久久久久| 免费国产99久久久香蕉| 欧美黑人激情性久久| 久久精品国产99久久久| 狠狠色综合久久久久尤物| 国产精品国色综合久久| 久久国产精品二国产精品| 精品久久久久久成人AV| 欧美大战日韩91综合一区婷婷久久青草| 久久只有这精品99| 亚洲国产成人乱码精品女人久久久不卡| 亚洲精品成人久久久| 国产精品免费看久久久| 免费久久人人爽人人爽av| 精品99久久aaa一级毛片| 久久香综合精品久久伊人| 精品久久亚洲中文无码| 亚洲人成电影网站久久| 免费国产99久久久香蕉| 国产Av激情久久无码天堂| 久久久久久免费视频| 精品水蜜桃久久久久久久| 国内精品久久久久影院免费| 久久99久国产麻精品66| 亚洲日韩欧美一区久久久久我| 精品久久综合1区2区3区激情| 久久国产色AV免费观看| 伊人久久大香线蕉亚洲五月天| 久久天天躁狠狠躁夜夜躁2014| 激情五月综合综合久久69| 9191精品国产免费久久| 99久久香蕉国产线看观香| 女同久久| 亚洲AV无一区二区三区久久| 精品一区二区久久久久久久网站|