Epoll 的一个巨坑!

多年前就用过epoll写socket服务,最终写出来的东西不稳定,即便用了libevent也同样会出现运行 不定时间之后出现能连接,但是服务器端毫无反应,客户端也不显示连接上,也不出现连接失败的一种情况。


今天由于需要想写个socket转发服务,再次拿起当年写的代码。


当一个人连接之后,我立即开一个线程,链接到转发服务器,然后把跟服务器的连接fd,和客户端的fd保存到一个struct,然后把这个传给event.data.ptr。


坑来了,event.data是个union,

如果这样写

event.data.ptr=&struct_data;

event.data.fd=client_fd;


在客户端有数据上来时,你读取不到event.data.ptr,因为

event.data.fd=client_fd;会覆盖event.data.ptr=&struct_data;。Unoin就是这样子的。


所以,只写,event.data.ptr=&struct_data;而client_fd保存到struct中


------------------

补充一个C语言的错误认识

int i;


typedef struct{
 int server_sock;
    int client_sock;
 int fd;
} final_client_data;

for(i=0;i<10;i++){

     final_client_data a;

     printf("%a\n", &a);

}

结果:

[root@server-164 proxy]# ./proxy 127.0.0.1 1234                                             
0x7fffa5204890
0x7fffa5204890
0x7fffa5204890
0x7fffa5204890
0x7fffa5204890
0x7fffa5204890
0x7fffa5204890
0x7fffa5204890
0x7fffa5204890
0x7fffa5204890

哭瞎了,,,,,



好的,这里是一个简单的 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 监听队列中,当某个连接的数据可读时,就读取并回显给客户端。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值