如何在C中编写基于epoll的套接字客户端

Jam*_*ead 1 c sockets client epoll

我在网上可以找到的所有示例都是服务器。我想使用epoll构建一个基本的Web搜寻器。因此,我需要一个基本的客户示例来入门。

当我说“基本”时,我实际上是指一个完整的示例,该示例演示了向实时Web主机发送和接收数据的多种连接。例如,一个简单的HEAD请求及其响应。

Mur*_*rli 5

这是带有epoll的客户端套接字的示例c代码。

#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 22
#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 = EPOLLIN; // 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, 1000/*timeout*/);
    for(i = 0; i < num_ready; i++) {
        if(events[i].events & EPOLLIN) {
            printf("Socket %d connected\n", events[i].data.fd);
        }
    }

    /*---Wait for data---*/
    num_ready = epoll_wait(epfd, events, MAX_EPOLL_EVENTS, 1000/*timeout*/);
    for(i = 0; i < num_ready; i++) {
        if(events[i].events & EPOLLIN) {
            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;
}
Run Code Online (Sandbox Code Playgroud)

  • 默认情况下,epoll 是级别触发的。对于它作为边缘触发需要或事件标志中的“| EPOLLET”位(event.events) (2认同)
  • 当您处理 100K 个对等点时,除了 epoll 接口之外,还有很多事情需要处理。例如,连接失败、重试、部分读取、缓慢对等、HTTP 解析错误等。它可以有多种不同的设计方式。通常,我会从一个用于连接到对等点的“连接”线程开始,以及一个用于从套接字读取数据的“读取”线程池。有一个类似 IPC 的管道,将连接的套接字 fd 从“连接”线程传递到“读取”线程。我确信当你开始使用它时,将会有更多这样的场景需要处理。一切顺利!! (2认同)
  • 对于每个套接字,应将带有EPOLL_CTL_ADD的epoll_ctl调用一次。当调用epoll_wait时,它将监视所有添加的套接字,并仅返回那些有一些事件要消耗的FD。 (2认同)