IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 网络协议 -> C 实现TCP服务端(select、poll、epoll) -> 正文阅读

[网络协议]C 实现TCP服务端(select、poll、epoll)

? ? ? ?使用C简单的实现一个tcp?server,包括常规server、多线程实现server、select实现server、poll实现server、epoll实现server。

????????IO模型原理可以看上一篇文章

常规模式

#define MAXLEN 4096

int main(int argc, char **argv) {
  int listenfd, connfd, n;
  struct sockaddr_in servaddr;
  char buff[MAXLEN];

  if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
    printf("create socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }

  memset(&servaddr, 0, sizeof(servaddr));
  servaddr.sin_family = AF_INET;
  servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  servaddr.sin_port = htons(9999);

  if (bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) == -1) {
    printf("bind socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }

  if (listen(listenfd, 10) != 0) {
    printf("listen socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }

  struct sockaddr_in client;
  socklen_t len = sizeof(client);
  if ((connfd = accept(listenfd, (struct sockaddr *)&client, &len)) == -1) {
    printf("accept socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  printf("========waiting for client's request========\n");
  while (1) {
    n = recv(connfd, buff, MAXLEN, 0);
    if (n > 0) {
      buff[n] = '\0';
      printf("recv msg from client: %s\n", buff);

      send(connfd, buff, n, 0);
    } else if (n == 0) {
      close(connfd);
    }

  close(listenfd);
  return 0;
}

?多线程模式

? ? ? ? 支持多个客户端并发连接。

#define MAXLEN 4096

void *client_routine(void *arg) {
  int connfd = *(int *)arg;
  char buff[MAXLEN];
  while (1) {
    int n = recv(connfd, buff, MAXLEN, 0);
    if (n > 0) {
      buff[n] = '\0';
      printf("recv msg from client: %s\n", buff);
      send(connfd, buff, n, 0);
    } else if (n == 0) {
      close(connfd);
      break;
    }
  }
  return NULL;
}

int main(int argc, char **argv) {
  int listenfd, connfd, n;
  struct sockaddr_in servaddr;
  char buff[MAXLEN];

  if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
    printf("create socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  memset(&servaddr, 0, sizeof(servaddr));
  servaddr.sin_family = AF_INET;
  servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  servaddr.sin_port = htons(9999);

  if (bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) == -1) {
    printf("bind socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  if (listen(listenfd, 10) != 0) {
    printf("listen socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  while (1) {
    struct sockaddr_in client;
    socklen_t len = sizeof(client);
    if ((connfd = accept(listenfd, (struct sockaddr *)&client, &len)) == -1) {
      printf("accept socket error: %s(errno: %d)\n", strerror(errno), errno);
      return 0;
    }
    pthread_t threadid;
    pthread_create(&threadid, NULL, client_routine, (void *)&connfd);
  }
  close(listenfd);
  return 0;
}

select模式

? ? ? ? 单线程支持多个客户端连接。

#define MAXLEN 4096

void *client_routine(void *arg) {
  int connfd = *(int *)arg;
  char buff[MAXLEN];
  while (1) {
    int n = recv(connfd, buff, MAXLEN, 0);
    if (n > 0) {
      buff[n] = '\0';
      printf("recv msg from client: %s\n", buff);
      send(connfd, buff, n, 0);
    } else if (n == 0) {
      close(connfd);
      break;
    }
  }
  return NULL;
}

int main(int argc, char **argv) {
  int listenfd, connfd, n;
  struct sockaddr_in servaddr;
  char buff[MAXLEN];

  if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
    printf("create socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  memset(&servaddr, 0, sizeof(servaddr));
  servaddr.sin_family = AF_INET;
  servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  servaddr.sin_port = htons(9999);

  if (bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) == -1) {
    printf("bind socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  if (listen(listenfd, 10) != 0) {
    printf("listen socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
fd_set rfds, rset, wfds, wset;
  FD_ZERO(&rfds);          /*将rfds清零使集合中不含任何fd*/
  FD_SET(listenfd, &rfds); /*将fd加入rfds集合*/

  FD_ZERO(&wfds);

  int max_fd = listenfd;

  while (1) {
    rset = rfds;
    wset = wfds;

    int nready = select(max_fd + 1, &rset, &wset, NULL, NULL);

    /*在调用select()函数后,用FD_ISSET来检测fd是否在set集合中,当检测到fd在set中则返回真,否则,返回假(0)*/
    if (FD_ISSET(listenfd, &rset)) {
      struct sockaddr_in client;
      socklen_t len = sizeof(client);
      if ((connfd = accept(listenfd, (struct sockaddr *)&client, &len)) == 1) {
        printf("accept socket error: %s(errno: %d)\n", strerror(errno), errno);
        return 0;
      }

      FD_SET(connfd, &rfds);

      if (connfd > max_fd) max_fd = connfd;

      if (--nready == 0) continue;
    }

    int i = 0;
    for (i = listenfd + 1; i <= max_fd; i++) {
      if (FD_ISSET(i, &rset)) {
        n = recv(i, buff, MAXLEN, 0);
        if (n > 0) {
          buff[n] = '\0';
          printf("recv msg from client: %s\n", buff);

          FD_SET(i, &wfds);
        } else if (n == 0) {
          FD_CLR(i, &rfds); /*将fd从set集合中清除*/
          close(i);
        }
        if (--nready == 0) break;
      } else if (FD_ISSET(i, &wset)) {
        send(i, buff, n, 0);
        FD_SET(i, &rfds);
        FD_CLR(i, &wfds); /*将fd从wfds集合中清除*/
      }
    }
  }
  close(listenfd);
  return 0;
}

?

poll 模式

#define MAXLEN 4096

void *client_routine(void *arg) {
  int connfd = *(int *)arg;
  char buff[MAXLEN];
  while (1) {
    int n = recv(connfd, buff, MAXLEN, 0);
    if (n > 0) {
      buff[n] = '\0';
      printf("recv msg from client: %s\n", buff);
      send(connfd, buff, n, 0);
    } else if (n == 0) {
      close(connfd);
      break;
    }
  }
  return NULL;
}

int main(int argc, char **argv) {
  int listenfd, connfd, n;
  struct sockaddr_in servaddr;
  char buff[MAXLEN];

  if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
    printf("create socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  memset(&servaddr, 0, sizeof(servaddr));
  servaddr.sin_family = AF_INET;
  servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  servaddr.sin_port = htons(9999);

  if (bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) == -1) {
    printf("bind socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  if (listen(listenfd, 10) != 0) {
    printf("listen socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
struct pollfd fds[POLL_SIZE] = {0};
  fds[0].fd = listenfd;
  fds[0].events = POLLIN;

  int max_fd = listenfd;
  int i = 0;
  for (i = 1; i < POLL_SIZE; i++) {
    fds[i].fd = -1;
  }

  while (1) {
    int nready = poll(fds, max_fd + 1, -1);
    printf("nready: %d \n", nready);
    if (fds[0].revents & POLLIN) {
      struct sockaddr_in client;
      socklen_t len = sizeof(client);
      if ((connfd = accept(listenfd, (struct sockaddr *)&client, &len)) == -1) {
        printf("accept socket error: %s(errno: %d)\n", strerror(errno), errno);
        return 0;
      }

      printf("accept \n");
      fds[connfd].fd = connfd;
      fds[connfd].events = POLLIN;

      if (connfd > max_fd) max_fd = connfd;

      if (--nready == 0) continue;
    }

    for (i = listenfd + 1; i <= max_fd; i++) {
      if (fds[i].revents & POLLIN) {
        n = recv(i, buff, MAXLEN, 0);
        if (n > 0) {
          buff[n] = '\0';
          printf("recv msg from client: %s\n", buff);
          fds[i].events = POLLOUT;
          // send(i, buff, n, 0);
        } else if (n == 0) {
          fds[i].fd = -1;
          close(i);
        }
        if (--nready == 0) break;
      } else if (fds[i].revents & POLLOUT) {
        send(i, buff, n, 0);
        fds[i].events = POLLIN;
      }
    }
  }
  close(listenfd);
  return 0;
}

epoll模式

#define MAXLEN 4096

void *client_routine(void *arg) {
  int connfd = *(int *)arg;
  char buff[MAXLEN];
  while (1) {
    int n = recv(connfd, buff, MAXLEN, 0);
    if (n > 0) {
      buff[n] = '\0';
      printf("recv msg from client: %s\n", buff);
      send(connfd, buff, n, 0);
    } else if (n == 0) {
      close(connfd);
      break;
    }
  }
  return NULL;
}

int main(int argc, char **argv) {
  int listenfd, connfd, n;
  struct sockaddr_in servaddr;
  char buff[MAXLEN];

  if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
    printf("create socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  memset(&servaddr, 0, sizeof(servaddr));
  servaddr.sin_family = AF_INET;
  servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  servaddr.sin_port = htons(9999);

  if (bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) == -1) {
    printf("bind socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  if (listen(listenfd, 10) != 0) {
    printf("listen socket error: %s(errno: %d)\n", strerror(errno), errno);
    return 0;
  }
  int epfd = epoll_create(1);  // init size

  struct epoll_event events[POLL_SIZE] = {0};
  struct epoll_event ev;

  ev.events = EPOLLIN;
  ev.data.fd = listenfd;

  epoll_ctl(epfd, EPOLL_CTL_ADD, listenfd, &ev);

  while (1) {
    int nready = epoll_wait(epfd, events, POLL_SIZE, 5);
    if (nready == -1) {
      continue;
    }

    int i = 0;
    for (i = 0; i < nready; i++) {
      int clientfd = events[i].data.fd;
      if (clientfd == listenfd) {
        struct sockaddr_in client;
        socklen_t len = sizeof(client);
        if ((connfd = accept(listenfd, (struct sockaddr *)&client, &len)) ==
            -1) {
          printf("accept socket error: %s(errno: %d)\n", strerror(errno),
                 errno);
          return 0;
        }
        printf("accept\n");
        ev.events = EPOLLIN;
        ev.data.fd = connfd;
        epoll_ctl(epfd, EPOLL_CTL_ADD, connfd, &ev);
      } else if (events[i].events & EPOLLIN) {
        n = recv(clientfd, buff, MAXLEN, 0);
        if (n > 0) {
          buff[n] = '\0';
          printf("recv msg from client: %s\n", buff);

          send(clientfd, buff, n, 0);
        } else if (n == 0) {
          ev.events = EPOLLIN;
          ev.data.fd = clientfd;

          epoll_ctl(epfd, EPOLL_CTL_DEL, clientfd, &ev);
          close(clientfd);
        }
      }
    }
  }
  close(listenfd);
  return 0;
}

  网络协议 最新文章
使用Easyswoole 搭建简单的Websoket服务
常见的数据通信方式有哪些?
Openssl 1024bit RSA算法---公私钥获取和处
HTTPS协议的密钥交换流程
《小白WEB安全入门》03. 漏洞篇
HttpRunner4.x 安装与使用
2021-07-04
手写RPC学习笔记
K8S高可用版本部署
mySQL计算IP地址范围
上一篇文章      下一篇文章      查看所有文章
加:2021-12-24 18:52:13  更:2021-12-24 18:52:36 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年10日历 -2024/10/5 7:51:56-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码