Reactor模式

文章轉自這裏,侵刪

Reactor模式

最近一直在看遊雙的《高性能linux服務器編程》一書,下載鏈接: http://download.csdn.net/detail/analogous_love/9673008

書上是這麼介紹Reactor模式的:
這裏寫圖片描述

/**  
 *@desc:   用reactor模式練習服務器程序,main.cpp 
 *@author: zhangyl 
 *@date:   2016.11.23 
 */  

#include <iostream>  
#include <string.h>  
#include <sys/types.h>  
#include <sys/socket.h>  
#include <netinet/in.h>  
#include <arpa/inet.h>  //for htonl() and htons()  
#include <unistd.h>  
#include <fcntl.h>  
#include <sys/epoll.h>  
#include <signal.h>     //for signal()  
#include <pthread.h>  
#include <semaphore.h>  
#include <list>  
#include <errno.h>  
#include <time.h>  
#include <sstream>  
#include <iomanip> //for std::setw()/setfill()  
#include <stdlib.h>  


#define WORKER_THREAD_NUM   5  

#define min(a, b) ((a <= b) ? (a) : (b))   

int g_epollfd = 0;  
bool g_bStop = false;  
int g_listenfd = 0;  
pthread_t g_acceptthreadid = 0;  
pthread_t g_threadid[WORKER_THREAD_NUM] = { 0 };  
pthread_cond_t g_acceptcond;  
pthread_mutex_t g_acceptmutex;  

pthread_cond_t g_cond /*= PTHREAD_COND_INITIALIZER*/;  
pthread_mutex_t g_mutex /*= PTHREAD_MUTEX_INITIALIZER*/;  

pthread_mutex_t g_clientmutex;  

std::list<int> g_listClients;  

void prog_exit(int signo)  
{  
    ::signal(SIGINT, SIG_IGN);  
    //::signal(SIGKILL, SIG_IGN);//該信號不能被阻塞、處理或者忽略  
    ::signal(SIGTERM, SIG_IGN);  

    std::cout << "program recv signal " << signo << " to exit." << std::endl;  

    g_bStop = true;  

    ::epoll_ctl(g_epollfd, EPOLL_CTL_DEL, g_listenfd, NULL);  

    //TODO: 是否需要先調用shutdown()一下?  
    ::shutdown(g_listenfd, SHUT_RDWR);  
    ::close(g_listenfd);  
    ::close(g_epollfd);  

    ::pthread_cond_destroy(&g_acceptcond);  
    ::pthread_mutex_destroy(&g_acceptmutex);  

    ::pthread_cond_destroy(&g_cond);  
    ::pthread_mutex_destroy(&g_mutex);  

    ::pthread_mutex_destroy(&g_clientmutex);  
}  

bool create_server_listener(const char* ip, short port)  
{  
    g_listenfd = ::socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK, 0);  
    if (g_listenfd == -1)  
        return false;  

    int on = 1;  
    ::setsockopt(g_listenfd, SOL_SOCKET, SO_REUSEADDR, (char *)&on, sizeof(on));  
    ::setsockopt(g_listenfd, SOL_SOCKET, SO_REUSEPORT, (char *)&on, sizeof(on));  

    struct sockaddr_in servaddr;  
    memset(&servaddr, 0, sizeof(servaddr));   
    servaddr.sin_family = AF_INET;  
    servaddr.sin_addr.s_addr = inet_addr(ip);  
    servaddr.sin_port = htons(port);  
    if (::bind(g_listenfd, (sockaddr *)&servaddr, sizeof(servaddr)) == -1)  
        return false;  

    if (::listen(g_listenfd, 50) == -1)  
        return false;  

    g_epollfd = ::epoll_create(1);  
    if (g_epollfd == -1)  
        return false;  

    struct epoll_event e;  
    memset(&e, 0, sizeof(e));  
    e.events = EPOLLIN | EPOLLRDHUP;  
    e.data.fd = g_listenfd;  
    if (::epoll_ctl(g_epollfd, EPOLL_CTL_ADD, g_listenfd, &e) == -1)  
        return false;  

    return true;  
}  

void release_client(int clientfd)  
{  
    if (::epoll_ctl(g_epollfd, EPOLL_CTL_DEL, clientfd, NULL) == -1)  
        std::cout << "release client socket failed as call epoll_ctl failed" << std::endl;  

    ::close(clientfd);  
}  

void* accept_thread_func(void* arg)  
{     
    while (!g_bStop)  
    {  
        ::pthread_mutex_lock(&g_acceptmutex);  
        ::pthread_cond_wait(&g_acceptcond, &g_acceptmutex);  
        //::pthread_mutex_lock(&g_acceptmutex);  

        //std::cout << "run loop in accept_thread_func" << std::endl;  

        struct sockaddr_in clientaddr;  
        socklen_t addrlen;  
        int newfd = ::accept(g_listenfd, (struct sockaddr *)&clientaddr, &addrlen);  
        ::pthread_mutex_unlock(&g_acceptmutex);  
        if (newfd == -1)  
            continue;  

        std::cout << "new client connected: " << ::inet_ntoa(clientaddr.sin_addr) << ":" << ::ntohs(clientaddr.sin_port) << std::endl;  

        //將新socket設置爲non-blocking  
        int oldflag = ::fcntl(newfd, F_GETFL, 0);  
        int newflag = oldflag | O_NONBLOCK;  
        if (::fcntl(newfd, F_SETFL, newflag) == -1)  
        {  
            std::cout << "fcntl error, oldflag =" << oldflag << ", newflag = " << newflag << std::endl;  
            continue;  
        }  

        struct epoll_event e;  
        memset(&e, 0, sizeof(e));  
        e.events = EPOLLIN | EPOLLRDHUP | EPOLLET;  
        e.data.fd = newfd;  
        if (::epoll_ctl(g_epollfd, EPOLL_CTL_ADD, newfd, &e) == -1)  
        {  
            std::cout << "epoll_ctl error, fd =" << newfd << std::endl;  
        }  
    }  

    return NULL;  
}  


void* worker_thread_func(void* arg)  
{     
    while (!g_bStop)  
    {  
        int clientfd;  
        ::pthread_mutex_lock(&g_clientmutex);  
        while (g_listClients.empty())  
            ::pthread_cond_wait(&g_cond, &g_clientmutex);  
        clientfd = g_listClients.front();  
        g_listClients.pop_front();    
        pthread_mutex_unlock(&g_clientmutex);  

        //gdb調試時不能實時刷新標準輸出,用這個函數刷新標準輸出,使信息在屏幕上實時顯示出來  
        std::cout << std::endl;  

        std::string strclientmsg;  
        char buff[256];  
        bool bError = false;  
        while (true)  
        {  
            memset(buff, 0, sizeof(buff));  
            int nRecv = ::recv(clientfd, buff, 256, 0);  
            if (nRecv == -1)  
            {  
                if (errno == EWOULDBLOCK)  
                    break;  
                else  
                {  
                    std::cout << "recv error, client disconnected, fd = " << clientfd << std::endl;  
                    release_client(clientfd);  
                    bError = true;  
                    break;  
                }  

            }  
            //對端關閉了socket,這端也關閉。  
            else if (nRecv == 0)  
            {  
                std::cout << "peer closed, client disconnected, fd = " << clientfd << std::endl;  
                release_client(clientfd);  
                bError = true;  
                break;  
            }  

            strclientmsg += buff;  
        }  

        //出錯了,就不要再繼續往下執行了  
        if (bError)  
            continue;  

        std::cout << "client msg: " << strclientmsg;  

        //將消息加上時間標籤後發回  
        time_t now = time(NULL);  
        struct tm* nowstr = localtime(&now);  
        std::ostringstream ostimestr;  
        ostimestr << "[" << nowstr->tm_year + 1900 << "-"   
                  << std::setw(2) << std::setfill('0') << nowstr->tm_mon + 1 << "-"   
                  << std::setw(2) << std::setfill('0') << nowstr->tm_mday << " "  
                  << std::setw(2) << std::setfill('0') << nowstr->tm_hour << ":"   
                  << std::setw(2) << std::setfill('0') << nowstr->tm_min << ":"   
                  << std::setw(2) << std::setfill('0') << nowstr->tm_sec << "]server reply: ";  

        strclientmsg.insert(0, ostimestr.str());  

        while (true)  
        {  
            int nSent = ::send(clientfd, strclientmsg.c_str(), strclientmsg.length(), 0);  
            if (nSent == -1)  
            {  
                if (errno == EWOULDBLOCK)  
                {  
                    ::sleep(10);  
                    continue;  
                }  
                else  
                {  
                    std::cout << "send error, fd = " << clientfd << std::endl;  
                    release_client(clientfd);  
                    break;  
                }  

            }            

            std::cout << "send: " << strclientmsg;  
            strclientmsg.erase(0, nSent);  

            if (strclientmsg.empty())  
                break;  
        }  
    }  

    return NULL;  
}  

void daemon_run()  
{  
    int pid;  
    signal(SIGCHLD, SIG_IGN);  
    //1)在父進程中,fork返回新創建子進程的進程ID;  
    //2)在子進程中,fork返回0;  
    //3)如果出現錯誤,fork返回一個負值;  
    pid = fork();  
    if (pid < 0)  
    {  
        std:: cout << "fork error" << std::endl;  
        exit(-1);  
    }  
    //父進程退出,子進程獨立運行  
    else if (pid > 0) {  
        exit(0);  
    }  
    //之前parent和child運行在同一個session裏,parent是會話(session)的領頭進程,  
    //parent進程作爲會話的領頭進程,如果exit結束執行的話,那麼子進程會成爲孤兒進程,並被init收養。  
    //執行setsid()之後,child將重新獲得一個新的會話(session)id。  
    //這時parent退出之後,將不會影響到child了。  
    setsid();  
    int fd;  
    fd = open("/dev/null", O_RDWR, 0);  
    if (fd != -1)  
    {  
        dup2(fd, STDIN_FILENO);  
        dup2(fd, STDOUT_FILENO);  
        dup2(fd, STDERR_FILENO);  
    }  
    if (fd > 2)  
        close(fd);  

}  


int main(int argc, char* argv[])  
{    
    short port = 0;  
    int ch;  
    bool bdaemon = false;  
    while ((ch = getopt(argc, argv, "p:d")) != -1)  
    {  
        switch (ch)  
        {  
        case 'd':  
            bdaemon = true;  
            break;  
        case 'p':  
            port = atol(optarg);  
            break;  
        }  
    }  

    if (bdaemon)  
        daemon_run();  


    if (port == 0)  
        port = 12345;  

    if (!create_server_listener("0.0.0.0", port))  
    {  
        std::cout << "Unable to create listen server: ip=0.0.0.0, port=" << port << "." << std::endl;  
        return -1;  
    }  


    //設置信號處理  
    signal(SIGCHLD, SIG_DFL);  
    signal(SIGPIPE, SIG_IGN);  
    signal(SIGINT, prog_exit);  
    //signal(SIGKILL, prog_exit);<span style="font-family: Arial, Helvetica, sans-serif;">//該信號不能被阻塞、處理或者忽略</span>  

    signal(SIGTERM, prog_exit);  

    ::pthread_cond_init(&g_acceptcond, NULL);  
    ::pthread_mutex_init(&g_acceptmutex, NULL);  

    ::pthread_cond_init(&g_cond, NULL);  
    ::pthread_mutex_init(&g_mutex, NULL);  

    ::pthread_mutex_init(&g_clientmutex, NULL);  

    ::pthread_create(&g_acceptthreadid, NULL, accept_thread_func, NULL);  
    //啓動工作線程  
    for (int i = 0; i < WORKER_THREAD_NUM; ++i)  
    {  
        ::pthread_create(&g_threadid[i], NULL, worker_thread_func, NULL);  
    }  

    while (!g_bStop)  
    {         
        struct epoll_event ev[1024];  
        int n = ::epoll_wait(g_epollfd, ev, 1024, 10);  
        if (n == 0)  
            continue;  
        else if (n < 0)  
        {  
            std::cout << "epoll_wait error" << std::endl;  
            continue;  
        }  

        int m = min(n, 1024);  
        for (int i = 0; i < m; ++i)  
        {  
            //通知接收連接線程接收新連接  
            if (ev[i].data.fd == g_listenfd)  
                pthread_cond_signal(&g_acceptcond);  
            //通知普通工作線程接收數據  
            else  
            {                 
                pthread_mutex_lock(&g_clientmutex);                
                g_listClients.push_back(ev[i].data.fd);  
                pthread_mutex_unlock(&g_clientmutex);  
                pthread_cond_signal(&g_cond);  
                //std::cout << "signal" << std::endl;  
            }  

        }  

    }  

    return 0;  
} 

程序的功能一個簡單的echo服務:客戶端連接上服務器之後,給服務器發送信息,服務器加上時間戳等信息後返回給客戶端。

程序具體解釋請參照原作者。

發佈了25 篇原創文章 · 獲贊 34 · 訪問量 2萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章