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

               C++ 技術中心

               :: 首頁 :: 聯系 ::  :: 管理
              160 Posts :: 0 Stories :: 87 Comments :: 0 Trackbacks

            公告

            鄭重聲明:本BLOG所發表的原創文章,作者保留一切權利。必須經過作者本人同意后方可轉載,并注名作者(天空)和出處(CppBlog.com)。作者Email:coder@luckcoder.com

            留言簿(27)

            搜索

            •  

            最新隨筆

            最新評論

            評論排行榜

            epoll是linux下高并發服務器的完美方案,因為是基于事件觸發的,所以比select快的不只是一個數量級。
            單線程epoll,觸發量可達到15000,但是加上業務后,因為大多數業務都與數據庫打交道,所以就會存在阻塞的情況,這個時候就必須用多線程來提速。
            業務在線程池內,這里要加鎖才行。測試結果2300個/s
            測試工具:stressmark
            因為加了適用與ab的代碼,所以也可以適用ab進行壓力測試。
            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(socketfd,buf, strlen(buf),0);
            #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 2013-11-26 15:49 C++技術中心 閱讀(5749) 評論(2)  編輯 收藏 引用 所屬分類: Linux 編程

            Feedback

            # re: 高并發的epoll+線程池,業務在線程池內[未登錄] 2013-11-27 12:05 春秋十二月
            這個例子實現描述的epoll+theadpool方案不太好
            1)fd讀事件用加鎖的queue通知,造成epoll thead和read thread間的同步,及many read threads間的鎖競爭
            2)fd寫事件沒有處理,后面的send(fd,buf,strlen(buf),0)也沒處理好  回復  更多評論
              

            # re: 高并發的epoll+線程池,業務在線程池內 2013-11-27 18:34 老爺
            大哥, 你可以封裝下啊, 你不覺得看起來挺費勁嗎? 看起來費勁, 用起來就更不用說了~  回復  更多評論
              

            久久婷婷人人澡人人| 大美女久久久久久j久久| 久久se精品一区精品二区国产| 亚洲国产精品无码久久久不卡| 国产免费久久久久久无码| 久久综合九色综合欧美狠狠| 亚洲午夜久久久久妓女影院| 中文字幕精品久久| 2020国产成人久久精品| 亚洲&#228;v永久无码精品天堂久久| 99久久婷婷国产一区二区| 青青青国产成人久久111网站| 久久96国产精品久久久| 久久综合丝袜日本网| 亚洲午夜精品久久久久久人妖| 青青青青久久精品国产| 久久精品这里热有精品| 九九久久精品国产| 青青草国产97免久久费观看| 久久国产欧美日韩精品免费| 午夜精品久久影院蜜桃| 久久久久久国产精品美女| 国内精品综合久久久40p| 久久男人Av资源网站无码软件| 天天躁日日躁狠狠久久| 国产美女久久久| 欧美与黑人午夜性猛交久久久| 国产欧美久久久精品影院| 久久久久免费看成人影片| 2020最新久久久视精品爱| 亚洲午夜无码AV毛片久久| 亚洲AV无码1区2区久久| 久久久精品一区二区三区| 无码人妻少妇久久中文字幕| 一本久久a久久精品亚洲| 久久久久久狠狠丁香| 亚洲国产成人精品无码久久久久久综合 | 久久久久亚洲av成人无码电影| 性高湖久久久久久久久AAAAA| 久久亚洲精品中文字幕| 久久99国产精品成人欧美|