【发布时间】:2025-11-26 00:40:02
【问题描述】:
我正在尝试使用 epoll 创建非阻塞连接。我还在学习 epoll,所以我尝试使用本示例中提到的代码 - How to code an epoll based sockets client in C。
我刚刚修改了超时并将事件设为 EPOLLOUT。
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <sys/socket.h>
#include <resolv.h>
#include <sys/epoll.h>
#include <arpa/inet.h>
#include <unistd.h>
#define PORT 20000
#define SERVER "127.0.0.1"
#define MAXBUF 1024
#define MAX_EPOLL_EVENTS 64
int main() {
int sockfd;
struct sockaddr_in dest;
char buffer[MAXBUF];
struct epoll_event events[MAX_EPOLL_EVENTS];
int i, num_ready;
/*---Open socket for streaming---*/
if ( (sockfd = socket(AF_INET, SOCK_STREAM|SOCK_NONBLOCK, 0)) < 0 ) {
perror("Socket");
exit(errno);
}
/*---Add socket to epoll---*/
int epfd = epoll_create(1);
struct epoll_event event;
event.events = EPOLLOUT; // Cann append "|EPOLLOUT" for write events as well
event.data.fd = sockfd;
epoll_ctl(epfd, EPOLL_CTL_ADD, sockfd, &event);
/*---Initialize server address/port struct---*/
bzero(&dest, sizeof(dest));
dest.sin_family = AF_INET;
dest.sin_port = htons(PORT);
if ( inet_pton(AF_INET, SERVER, &dest.sin_addr.s_addr) == 0 ) {
perror(SERVER);
exit(errno);
}
/*---Connect to server---*/
if ( connect(sockfd, (struct sockaddr*)&dest, sizeof(dest)) != 0 ) {
if(errno != EINPROGRESS) {
perror("Connect ");
exit(errno);
}
}
/*---Wait for socket connect to complete---*/
num_ready = epoll_wait(epfd, events, MAX_EPOLL_EVENTS, 100000/*timeout*/);
for(i = 0; i < num_ready; i++) {
if(events[i].events & EPOLLOUT) {
printf("Socket %d connected\n", events[i].data.fd);
}
}
/*---Wait for data---*/
num_ready = epoll_wait(epfd, events, MAX_EPOLL_EVENTS, 100000);
for(i = 0; i < num_ready; i++) {
if(events[i].events & EPOLLOUT) {
printf("Socket %d got some data\n", events[i].data.fd);
//bzero(buffer, MAXBUF);
//recv(sockfd, buffer, sizeof(buffer), 0);
//printf("Received: %s", buffer);
}
}
close(sockfd);
return 0;
}
结果:
[root@test-vm socket_prgm]# ./test_epoll_so.o
Socket 3 connected
Socket 3 got some data
我在上述程序中观察到的两件事:
- 在端口 20000 上没有运行服务器进程。套接字仍然能够连接事件 EPOLLOUT。
- 超时期限根本不起作用。脚本立即执行。
我在这里遗漏了一些明显的东西吗?谢谢你。
【问题讨论】:
-
建议检查来自
epoll_ctl()的返回值,以确保它按照您的想法进行操作。 -
@SteveFried
epoll_ctl()的返回值为0。我希望这是在做预期的事情。 -
尝试通过删除
SOCK_NONBLOCK来阻塞套接字。 -
@tambre 现在输出是“连接被拒绝”。我认为这可能是因为该端口上没有运行守护程序。我尝试使用端口 22,我得到了与上面相同的结果,但没有等待超时时间。另外,我希望它用于非阻塞套接字。我可以在许多示例中看到,其中大多数在 epoll_wait 中将 -1(应该无限期阻塞)作为超时,但它将在
while true or for(;;)循环内。 -
你忽略了
EINPROGRESS为connect()。如果套接字没有连接或绑定,大概 poll 会立即返回。