关于epoll的调试的几个问题

将今天调试的几个小问题点总结下,后续遇到再添加。

一 将总结的问题点放在最前面 

1 epoll_wait的maxevents参数

epoll_wait的maxevents参数,经过测试,maxevents的值可以小于监听的描述符总数。我在下面的测试代码中,将这个值改为1,也是可以同时处理两个客户端的。

2 删除文件事件集合中的文件描述符

如下所示,最后一个参数设置为NULL即可;

ret = epoll_ctl(epfd, EPOLL_CTL_DEL,sockfd, NULL);

设置为下面的形式,就是最后一个参数配置为何EPOLL_CTL_ADD时相同也没问题:

ev.events = EPOLLIN | EPOLLET;
ev.data.fd = sockfd;
ret = epoll_ctl(epfd, EPOLL_CTL_DEL,sockfd, &ev);

3 EPOLL_CTL_DEL调用要在close之后

如果调用EPOLL_CTL_DEL之前先调用close。就会报错:

EPOLL_CTL_DEL: Bad file descriptor

参考资料:《UNIX系统编程手册 下》 

 

二 基础相关

相关结构体:

     typedef union epoll_data {
        void *ptr;
        int fd;
        __uint32_t u32;
        __uint64_t u64;
     } epoll_data_t;
      //感兴趣的事件和被触发的事件
     struct epoll_event {
        __uint32_t events; /* Epoll events */
        epoll_data_t data; /* User data variable */
     };

函数

     int epoll_create(int size);

 size只要是大于的1的数任意数都可以:老师说的。老师有理

     int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);

其中参数epfd表示要操作的文件描述符,它是epoll_create的返回值;第二个参数op表示动作,使用以下三个宏来表示:

· EPOLL_CTL_ADD:注册新的fd到epfd中。

· EPOLL_CTL_MOD:修改已经注册的fd的监听事件。

· EPOLL_CTL_DEL:从epfd中删除一个fd。 

第三个参数fd是op实施的对象,即需要操作的文件描述符;第四个参数event是告诉内核需要监听什么事件,events可以是以下几个宏的集合:· EPOLLIN:表示对应的文件描述符可以读(包括对SOCKET正常关闭)。· EPOLLOUT:表示对应的文件描述符可以写。· EPOLLPRI:表示对应的文件描述符有紧急的数据可读(这里应该表示有带外数据到来)。· EPOLLERR:表示对应的文件描述符发生错误。· EPOLLHUP:表示对应的文件描述符被挂断。· EPOLLET:将EPOLL设为边缘触发(Edge Triggered)模式,这是相对于水平触发(Level Triggered)来说的。 · EPOLLONESHOT:只监听一次事件,当监听完这次事件之后,如果还需要继续监听这个socket的话,需要再次把这个socket加入到EPOLL队列里。

     #include <sys/epoll.h>
     int epoll_wait ( int epfd, struct epoll_event* events, int maxevents, int
timeout );

其中参数epfd表示要操作的文件描述符,它是epoll_create的返回值;events指向检测到的事件集合,将所有就绪的事件从内核事件表中复制到它的第二个参数events指向的数组中;maxevents指定最多监听多少个事件;timeout指定epoll的超时时间,单位是毫秒。当timeout设置为‒1时,epoll_wait调用将永远阻塞,直到某个事件发生;当timeout设置为0时,epoll_wait调用将立即返回;当timeout设置为大于0时,表示指定的毫秒。函数执行成功时返回就绪的文件描述符的个数,失败时返回‒1并设置errno。 

三 测试代码

#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>
#include <pthread.h>
#include <sys/select.h>
#include <sys/epoll.h>
#include <errno.h>
#define DEBUG_INFO(format, ...) printf("%s:%d -- " format "\n", __func__, __LINE__,##__VA_ARGS__)
#define BUFFER_SIZE 1024

void set_nonblock(int fd){
    int flag = fcntl(fd,F_GETFL);
    if(flag == -1){
        perror("set_nonblock");
        exit(1);
    }
    flag |= O_NONBLOCK;
    if(fcntl(fd,F_SETFL, flag) < 0){
        perror("fcntl");
        exit(1);
    }
}

void test_01(){
    int listenfd;
    int on = 1;
    struct sockaddr_in server_addr;

    listenfd = socket(AF_INET, SOCK_STREAM, 0);

    if(setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR,&on,sizeof(on)) < 0){
        perror("setsockopt");
        exit(-1);
    }
    DEBUG_INFO("begin bing");
    server_addr.sin_port = htons(9878);
    server_addr.sin_family = AF_INET;
    server_addr.sin_addr.s_addr = INADDR_ANY;
    if(bind(listenfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0){
        perror("bind");
        exit(0);
    }
    listen(listenfd,5);

    set_nonblock(listenfd);
    int epfd = epoll_create(2);
    if(epfd < 0){
        perror("epoll_create");
        exit(-1);
    }
    struct epoll_event ev;
    ev.events = EPOLLIN;
    ev.data.fd = listenfd;
    epoll_ctl(epfd, EPOLL_CTL_ADD,listenfd, &ev);

    // int epoll_wait(int epfd, struct epoll_event *events, int maxevents, int timeout);
    #define MAX_EVENT_NUMBER 1
    struct epoll_event events[MAX_EVENT_NUMBER];
    char buf[BUFFER_SIZE];
    DEBUG_INFO("epoll_wait");
    while(1){
        int ret = epoll_wait(epfd, events,MAX_EVENT_NUMBER,-1);
        for(int i = 0;i < ret;i++){
            int sockfd = events[i].data.fd;
            if(events[i].events&EPOLLIN && sockfd == listenfd){
                struct sockaddr_in clientaddr;
                socklen_t addrlen = sizeof(clientaddr);
                int clientfd = accept(listenfd,(struct sockaddr*)&clientaddr,&addrlen);
                DEBUG_INFO("clientfd = %d",clientfd);
                ev.events = EPOLLIN | EPOLLET;
                ev.data.fd = clientfd;
                epoll_ctl(epfd, EPOLL_CTL_ADD,clientfd, &ev);
                set_nonblock(clientfd);
            }else if(events[i].events&EPOLLIN){
                //memset(buf,0,sizeof(buf));
                bzero(buf,sizeof(buf));
                int index = 0;
                int readlen = 0;
                do{
                    readlen = recv(events[i].data.fd,&buf[index],1,0);
                    
                    if(readlen < 1){
                        DEBUG_INFO("readlen = %d",readlen);
                        break;
                    }else{
                        index++;
                    }
                }while(1);
                DEBUG_INFO("index = %d",index);
                int local_errno = errno;
                if(index > 0){
                    DEBUG_INFO("len = %d,buf = %s",index,buf);
                    int sendlen = send(sockfd,buf,index,0);
                    DEBUG_INFO("sendlen = %d",sendlen);
                }
                if(readlen == 0){
                    DEBUG_INFO("close %d",sockfd);
                    close(sockfd);
                    epoll_ctl(epfd, EPOLL_CTL_DEL,sockfd, &events[i]);
                }else if(readlen < 0){
                    if(local_errno == EINTR || local_errno == EWOULDBLOCK || local_errno == EAGAIN){
                        //
                        DEBUG_INFO("read finished");
                    }else{
                        perror("read");
                        DEBUG_INFO("close %d",sockfd);
                        close(sockfd);
                        epoll_ctl(epfd, EPOLL_CTL_DEL,sockfd, &events[i]);
                    }
                }                
            }else{
                DEBUG_INFO("unknown");
            }
        }
    }
    exit(0);
}

int main(int argc, char **argv)
{
    test_01();
}

CMakeLists.txt

project(mq_project)

cmake_minimum_required(VERSION 3.16)

SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11 -pthread")
# set(CMAKE_C_COMPILER /arm-gcc/bin/arm-linux-gnueabihf-gcc)
# set(CMAKE_CXX_COMPILER /arm-gcc/bin/arm-linux-gnueabihf-g++)
# add_executable(test test.cpp)
# add_executable(test posix_thread.cpp)
add_executable(test epoll.cpp)

小结

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
好的,这里是一个简单的 epoll 例子: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <errno.h> #include <sys/socket.h> #include <arpa/inet.h> #include <sys/epoll.h> #define MAX_EVENTS 10 int main(int argc, char *argv[]) { int listen_sock, conn_sock, epoll_fd, nfds, i, n; struct sockaddr_in serv_addr, cli_addr; socklen_t cli_len = sizeof(cli_addr); struct epoll_event ev, events[MAX_EVENTS]; char buffer[1024]; if ((listen_sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) { perror("socket"); exit(EXIT_FAILURE); } memset(&serv_addr, 0, sizeof(serv_addr)); serv_addr.sin_family = AF_INET; serv_addr.sin_addr.s_addr = htonl(INADDR_ANY); serv_addr.sin_port = htons(8080); if (bind(listen_sock, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0) { perror("bind"); exit(EXIT_FAILURE); } if (listen(listen_sock, SOMAXCONN) < 0) { perror("listen"); exit(EXIT_FAILURE); } if ((epoll_fd = epoll_create1(0)) < 0) { perror("epoll_create1"); exit(EXIT_FAILURE); } ev.events = EPOLLIN; ev.data.fd = listen_sock; if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, listen_sock, &ev) < 0) { perror("epoll_ctl: listen_sock"); exit(EXIT_FAILURE); } while (1) { nfds = epoll_wait(epoll_fd, events, MAX_EVENTS, -1); if (nfds < 0) { perror("epoll_wait"); exit(EXIT_FAILURE); } for (i = 0; i < nfds; ++i) { if (events[i].data.fd == listen_sock) { conn_sock = accept(listen_sock, (struct sockaddr *) &cli_addr, &cli_len); if (conn_sock < 0) { perror("accept"); exit(EXIT_FAILURE); } ev.events = EPOLLIN; ev.data.fd = conn_sock; if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, conn_sock, &ev) < 0) { perror("epoll_ctl: conn_sock"); exit(EXIT_FAILURE); } } else { n = read(events[i].data.fd, buffer, sizeof(buffer)); if (n < 0) { perror("read"); exit(EXIT_FAILURE); } if (n == 0) { if (epoll_ctl(epoll_fd, EPOLL_CTL_DEL, events[i].data.fd, NULL) < 0) { perror("epoll_ctl: conn_sock"); exit(EXIT_FAILURE); } close(events[i].data.fd); } else { if (write(events[i].data.fd, buffer, n) < 0) { perror("write"); exit(EXIT_FAILURE); } } } } } return 0; } ``` 这个程序监听端口 8080,并使用 epoll 实现了一个简单的回显服务器。它可以同时处理多个连接,当有新连接到来时,将其加入 epoll 监听队列中,当某个连接的数据可读时,就读取并回显给客户端。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

千册

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值