IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 网络协议 -> epoll实现udp并发 -> 正文阅读

[网络协议]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;
}

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

  网络协议 最新文章
使用Easyswoole 搭建简单的Websoket服务
常见的数据通信方式有哪些?
Openssl 1024bit RSA算法---公私钥获取和处
HTTPS协议的密钥交换流程
《小白WEB安全入门》03. 漏洞篇
HttpRunner4.x 安装与使用
2021-07-04
手写RPC学习笔记
K8S高可用版本部署
mySQL计算IP地址范围
上一篇文章      下一篇文章      查看所有文章
加:2021-09-26 10:33:32  更:2021-09-26 10:34:44 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/26 1:54:43-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码