Administrator
发布于 2022-03-11 / 7 阅读
0
0

epoll中 LT 和 ET 区别

epoll中 LT 和 ET 区别

上文解释过epoll 原理,现在梳理下epoll 的用法和 epoll 中两个读取数据的模式

现在用epoll 实现一个多路复用的服务器(代码在最后)

  • LT 水平触发 有数据到来就一直读,直到没有数据可以读取了
  • ET 边缘触发 有到无,读规定的字节数,没读完,下次有事件了再读取

读取客户端发来的数据时候,只读五个字节,看下ET和LT会有什么不同

!image-20220311222200962

  • ET模式下 必须使用非阻塞fd

!image-20220311222904069

连接后:

!image-20220311222323848

第一次客户端发了 7 个字节,而服务器只接受五个字节就结束,第二次客户端发送一个字节,客户读取了三个字节,显然还是读取上一次未读取的完的, 所以et模式就是 自己能在缓冲区读多少就读多少,读不完下次还来读

  • LT模式下:

如果没有设置,默认的是LT模式

!image-20220311222834196

!image-20220311223210774

客户端发送了15个字节,服务器读了三次,也就是说只要缓冲区没读完,就一直有事件响应, LT 等效于 (一个while + ET)

实现原理还有待后续挖掘(待填坑)

服务端代码:


#include <arpa/inet.h>
#include <errno.h>
#include <fcntl.h>
#include <netinet/tcp.h>
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/epoll.h>
#include <sys/poll.h>
#include <unistd.h>
#define BUFFER_LENGTH 1024
#define POLL_SIZE 1024
#define EPOLL_SIZE 1024

int main(int argc, char *argv[]) {
  if (argc < 2) {
    printf("Paramter Error\n");
    return -1;
  }
  int port = atoi(argv[1]);

  int sockfd = socket(AF_INET, SOCK_STREAM, 0);
  if (sockfd < 0) {
    perror("socket");
    return -1;
  }

  struct sockaddr_in addr;
  memset(&addr, 0, sizeof(struct sockaddr_in));

  addr.sin_family = AF_INET;
  addr.sin_port = htons(port);
  addr.sin_addr.s_addr = INADDR_ANY;

  if (bind(sockfd, (struct sockaddr *)&addr, sizeof(struct sockaddr_in)) < 0) {
    perror("bind");
    return 2;
  }

  if (listen(sockfd, 5) < 0) {
    perror("listen");
    return 3;
  }
  //相当于建立一颗  红黑树
  int epoll_fd = epoll_create(EPOLL_SIZE);
  // ev 是保存 fd的结构体
  struct epoll_event ev, events[EPOLL_SIZE] = {0};
  //初始化 ev
  ev.events = EPOLLIN;  //事件类型
  ev.data.fd = sockfd;  // sockfd是监听有无用户连接的io
  //向红黑树中插入 节点
  epoll_ctl(epoll_fd, EPOLL_CTL_ADD, sockfd, &ev);
  //开始
  while (1) {
    // nready 是意思是有几个事件产生,并且会将产生的事件放在 events数组中
    int nready = epoll_wait(epoll_fd, events, EPOLL_SIZE, -1);
    if (nready == -1) {
      printf("epoll_wait\n");
      break;
    }
    //开始遍历
    int i = 0;
    for (i = 0; i < nready; i++) {
      // sockfd有响应说明 有新的客户端来连接了
      if (events[i].data.fd == sockfd) {
        struct sockaddr_in client_addr;
        memset(&client_addr, 0, sizeof(struct sockaddr_in));
        socklen_t client_len = sizeof(client_addr);

        int clientfd =
            accept(sockfd, (struct sockaddr *)&client_addr, &client_len);
        if (clientfd <= 0) continue;

        char str[INET_ADDRSTRLEN] = {0};
        printf("recvived from %s at port %d, sockfd:%d, clientfd:%d\n",
               inet_ntop(AF_INET, &client_addr.sin_addr, str, sizeof(str)),
               ntohs(client_addr.sin_port), sockfd, clientfd);

        ev.events = EPOLLIN;
        ev.data.fd = clientfd;
        epoll_ctl(epoll_fd, EPOLL_CTL_ADD, clientfd, &ev);
      } else {  //这种情况是连接得用户有io响应
        int clientfd = events[i].data.fd;

        char buffer[BUFFER_LENGTH] = {0};
        //读取数据
        int ret = recv(clientfd, buffer, 5, 0);
        if (ret < 0) {
          if (errno == EAGAIN || errno == EWOULDBLOCK) {
            printf("read all data");
            continue;
          }
          //释放  fd
          close(clientfd);
		 //设置为  ET模式
          ev.events = EPOLLIN | EPOLLET;
          ev.data.fd = clientfd;
          //从红黑树删除
          epoll_ctl(epoll_fd, EPOLL_CTL_DEL, clientfd, &ev);
        } else if (ret == 0) {
          printf(" disconnect %d\n", clientfd);
          // ret == 0;  ret
          close(clientfd);

          ev.events = EPOLLIN;
          ev.data.fd = clientfd;
          epoll_ctl(epoll_fd, EPOLL_CTL_DEL, clientfd, &ev);

          break;
        } else {
          printf("Recv: %s, %d Bytes\n", buffer, ret);
        }
      }
    }
  }
  return 0;
}

客户代码:


#include <stdio.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <sys/types.h>
// struct sockaddr_in
#include <netinet/in.h>
// inet_addr
#include <arpa/inet.h>
#include <string.h>
#include <unistd.h>
#include<signal.h>
#include <iostream>
#include <string>
int x;
void handler(int s) //信号到来,则执行这个函数,输出超时
{
  close(x);
  exit(1);
}
int main() {
  int socket_desc;
  struct sockaddr_in server;
  char *message, server_reply[2000];
  signal(SIGINT, handler);
  // 创建socket
  socket_desc = socket(AF_INET, SOCK_STREAM, 0);
  if (-1 == socket_desc) {
    perror("cannot create socket");
    exit(1);
  }
  x = socket_desc;
  // 设置远程服务器的信息
  server.sin_addr.s_addr = inet_addr("127.0.0.1");
  server.sin_family = AF_INET;
  server.sin_port = htons(1234);

  // 连接
  if (connect(socket_desc, (struct sockaddr *)&server, sizeof(server)) < 0) {
    perror("cannot connect");
    return 0;
  }
  // 发送数据

  while (1) {
    std::string str;
    std::cin >> str;
    str.c_str();
    if (send(socket_desc, str.c_str(), str.size(), 0) < 0) {
      perror("send data error");
      return 2;
    }
    sleep(1);
  }
  printf("connect success");
  return 0;
}

\-


评论