epoll实现udp并发

#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/wait.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <fcntl.h>
#include <sys/epoll.h>
#include <sys/time.h>
#include <sys/resource.h>
#include <pthread.h>
#include <assert.h>
#include <list>
#define     HEADSIZE        5
#define     MAXEPOLLSIZE    20
#define     PORT            8888
#define    _PKG_INIT        0
#define    _PKG_RECVING     1
class subscriber
{
    
    
private:
    std::list<char *>       m_MsgRecvQueue;
    int listener, epoll_udp_fd;
    socklen_t len;
    struct sockaddr_in subscriber_addr;
    struct epoll_event event;
    struct epoll_event new_event;
    struct epoll_event events[MAXEPOLLSIZE];     //用于存放epoll_wait返回的可读事件
    struct sub_connection_s
    {
    
    
        sub_connection_s():fd(-1),irecvlen(0),curStat(_PKG_INIT),precvbuf(NULL),precvMemPointer(NULL){
    
    }
        ~sub_connection_s()
        {
    
    
            //delete []precvbuf;
        }
        int                   fd;
        int                   irecvlen;
        unsigned char         curStat;
        char                  dataHeadInfo[HEADSIZE];
        char                  *precvbuf;
        char                  *precvMemPointer;
    };
public:
    subscriber()
    {
    
    
        int ret = 0;
        int opt = 1;
        /*创建一个UDP套接字*/
        if((listener = socket(PF_INET, SOCK_DGRAM, 0)) == -1)
        {
    
    
            perror("socket");
            exit(1);
        }
        if((ret = setsockopt(listener, SOL_SOCKET, SO_REUSEADDR, &opt,sizeof(opt))) != 0)
        {
    
    
            exit(1);
        }
        if((ret = setsockopt(listener, SOL_SOCKET, SO_REUSEPORT, &opt, sizeof(opt))) !=0)
        {
    
    
            exit(1);
        }
        bzero(&subscriber_addr, sizeof(subscriber_addr));
        /*地址填充*/
        subscriber_addr.sin_family = PF_INET;
        subscriber_addr.sin_port = htons(PORT);
        subscriber_addr.sin_addr.s_addr = INADDR_ANY;
        /*将创建的UDP套接字和结构体地址绑定*/
        if(bind(listener, (struct sockaddr *)&subscriber_addr, sizeof(struct sockaddr)) == -1)
        {
    
    
            perror("bind");
            exit(1);
        }
    }

    bool epoll_udp_init()
    {
    
    
        /*创建一个epoll对象*/
        epoll_udp_fd = epoll_create(MAXEPOLLSIZE);
        /*关注可读事件,默认模式为LT模式*/
        event.events = EPOLLIN;
        /*套接字赋值*/
        event.data.fd = listener;
        /*将该事件作为一个节点加入到epoll对象中来*/
        if(epoll_ctl(epoll_udp_fd, EPOLL_CTL_ADD, listener, &event) < 0)
        {
    
    
            fprintf(stderr, "epoll set insertion error: fd=%dn", listener);
            return false;
        }
        return true;
    }

    int epoll_udp_accept(int recvsocket, struct sockaddr_in new_addr)
    {
    
    
        int new_sd = -1, ret = 0,reuse = 1;
        struct sockaddr_in peer_addr;
        socklen_t cli_len = sizeof(peer_addr);
        /*为新连接分配一个连接*/
        sub_connection_s *pConn = new sub_connection_s();
        memset(pConn, 0, sizeof(sub_connection_s));
        /*收包头*/
        if((ret = recvfrom(recvsocket, pConn->dataHeadInfo, HEADSIZE, 0, (struct sockaddr *)&peer_addr, &cli_len)) > 0)
        {
    
    
            printf("pConn->dataHeadInfo = %s\n", pConn->dataHeadInfo);
            printf("ret = %d\n", ret);
            if(ret == HEADSIZE)
            {
    
    
                pConn->irecvlen = atoi(pConn->dataHeadInfo);
                pConn->precvbuf = new char[pConn->irecvlen];
                pConn->precvMemPointer = pConn->precvbuf;
                printf("pConn->irecvlen = %d\n", pConn->irecvlen);
            }
        }
        else
        {
    
    
            printf("error \n");
        }
        /*创建一个UDP套接字*/
        if((new_sd = socket(PF_INET, SOCK_DGRAM, 0)) == -1)
        {
    
    
            perror("child socket");
            exit(1);
        }
        else
        {
    
    
            pConn->fd = new_sd;
        }
        /*设置地址复用功能*/
        if((ret = setsockopt(new_sd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse))) != 0)
        {
    
    
            exit(1);
        }
        /*设置端口复用功能*/
        if((ret = setsockopt(new_sd, SOL_SOCKET, SO_REUSEPORT, &reuse, sizeof(reuse))) != 0)
        {
    
    
            exit(1);
        }
        /*以一个新的套接字重新绑定这个客户端,后面接收数据就从这个新的套接字接收了*/
        if((ret = bind(new_sd, (struct sockaddr *)&new_addr, sizeof(struct sockaddr))) != 0)
        {
    
    
            perror("new bind");
            exit(1);
        }
        /*设置地址协议族*/
        peer_addr.sin_family = PF_INET;
        /*发起连接*/
        if(connect(new_sd, (struct sockaddr *) &peer_addr, sizeof(struct sockaddr)) == -1)
        {
    
    
            perror("new connect");
            exit(1);
        }
        /*关注可读事件*/
        new_event.events = EPOLLIN;
        /*为新接入的客户端增加文件描述符*/
        new_event.data.fd = new_sd;
        /*为每一个连接建立一个连接*/
        new_event.data.ptr = (void *)pConn;
        /*把新连接作为一个节点加到红黑树当中*/
        if(epoll_ctl(epoll_udp_fd, EPOLL_CTL_ADD, new_sd, &new_event) < 0)
        {
    
    
            fprintf(stderr, "epoll set insertion error: fd=%dn", new_sd);
            return -1;
        }
        return 1;
    }
    void epoll_udp_loop()
    {
    
    
        for(;;)
        {
    
    
            /*等待新的事件来*/
            int nfds = epoll_wait(epoll_udp_fd, events, MAXEPOLLSIZE, -1);
            if(nfds == -1)
            {
    
    
                perror("epoll_wait");
                break;
            }
            for(int n = 0; n < nfds; ++n)
            {
    
    
                if(events[n].data.fd == listener)
                {
    
    
                    printf("listener:%d\n", n);
                    epoll_udp_accept(listener, subscriber_addr);
                }
                else
                {
    
    
                    recv_data(events[n].data.ptr);
                }
            }
        }
    }
    bool recv_data(void *pTempConn)
    {
    
    
        int reco = 0;
        sub_connection_s *pConn = (sub_connection_s *)pTempConn;
        if((reco = recvfrom(pConn->fd, pConn->precvbuf, pConn->irecvlen, 0, NULL, NULL)) > 0)
        {
    
    
            printf("read[%d]: %s  from  %d\n", reco, pConn->precvbuf, pConn->fd);
            /*收包初始化状态*/
            if(pConn->curStat == _PKG_INIT)
            {
    
    
                /*收包完毕*/
                if(pConn->irecvlen == reco)
                {
    
    
                    /*将收到的数据包存储到队列中*/
                    printf("6666\n");
                    printf("pConn->precvMemPointer = %s\n", pConn->precvMemPointer);
                    m_MsgRecvQueue.push_back(pConn->precvMemPointer);
                    /*恢复初始状态*/
                    pConn->precvMemPointer = NULL;
                    pConn->curStat         = _PKG_INIT;
                    pConn->precvbuf        = NULL;
                    pConn->irecvlen        = 0;
                }
                else
                {
    
    
                    /*此时接收包头处于接收状态*/
                    pConn->curStat   = _PKG_RECVING;
                    /*收包缓冲区往后走*/
                    pConn->precvbuf  = pConn->precvbuf + reco;
                    /*剩余的要接收包头的长度*/
                    pConn->irecvlen  = pConn->irecvlen - reco;
                }
            }
            /*收包当中*/
            else if(pConn->curStat == _PKG_RECVING)
            {
    
    
                /*收包完毕*/
                if(pConn->irecvlen == reco)
                {
    
    
                    /*将收到的数据包存储到队列中*/
                    m_MsgRecvQueue.push_back(pConn->precvMemPointer);
                    printf("8888\n");
                    printf("pConn->precvMemPointer = %s\n", pConn->precvMemPointer);
                    /*恢复初始状态*/
                    pConn->precvMemPointer = NULL;
                    pConn->curStat         = _PKG_INIT;
                    pConn->precvbuf        = NULL;
                    pConn->irecvlen        = 0;
                }
                else
                {
    
    
                    /*收包缓冲区往后走*/
                    pConn->precvbuf  =  pConn->precvbuf + reco;
                    /*剩余的要接收包头的长度*/
                    pConn->irecvlen  =  pConn->irecvlen - reco;
                }
            }
        }
        return true;
    }
    ~subscriber()
    {
    
    
        close(listener);
    }
};

int main(int argc, char **argv)
{
    
    
    subscriber obj;
    obj.epoll_udp_init();
    obj.epoll_udp_loop();
    return 0;
}

在这里插入图片描述
在这里插入图片描述

猜你喜欢

转载自blog.csdn.net/qq_38158479/article/details/120464304