Linux C TCP服务器端-select案例

简介: 本文主要介绍了linux下Select的TCP通信流程,实现了客户端和服务器的通信,主要实现了消息的回发,即服务器将消息原封不动的回发给客户端。

前言

本文主要介绍了linux下Select的TCP通信流程,实现了客户端和服务器的通信,主要实现了消息的回发,即服务器将消息原封不动的回发给客户端。

这里主要介绍服务端代码,关于客户端代码请参考客户端代码和socket的基本使用


服务器端代码

#include <errno.h>
#include <netinet/in.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <unistd.h>
#define MAXLNE  4096
int main(int argc, char **argv) 
{
    int listenfd, connfd, n;
    struct sockaddr_in servaddr;
    char buff[MAXLNE];
    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;
    }
    //listen接口之后会一直监听客户端的连接,每次客户端连接,都会和其创建连接(三次连接时内核完成的,不是由应用程序去控制的)
  //三次握手不发生在任何API中,协议栈本身被动完成的。
    if (listen(listenfd, 10) == -1) {
        printf("listen socket error: %s(errno: %d)\n", strerror(errno), errno);
        return 0;
    }
    //select模型相比于一请求一线程而言
  //一个select最多支持1024 socket,如果开多个线程,那么是能支持大量socket的。
    //select不能做到100w级别的fd,select函数本身有copy全部监控的socket,循环这些
  //全部监控的socket存在一定的效率问题
  // fd_set,每个socket对应一个bit位,bit位为1,代表该socket发生了事件。
  //rfds:是需要监控读的集合(从用户copy到内核)
  //rset:是发生了读事件的socket的集合(从内核copy到用户)
  fd_set rfds, rset, wfds, wset;
  FD_ZERO(&rfds);//清空bit位
  FD_SET(listenfd, &rfds);//设置监听的socket的bit位
  FD_ZERO(&wfds);
    //比如listenfd为100,那么select就需要从第一位bit监控到101位的bit
  int max_fd = listenfd;//select第一个参数为监听socket的最大值+1
  while (1) {
  rset = rfds;
  wset = wfds;
        //一个专门的服务员来给所有的客户端服务,可以知道到底是哪些客户端发生了相应的事件(比如发送了数据给服务器)
  //如果最后一个参数为NULL,那么是阻塞的,一直到有事件发生(nready会大于0)
  int nready = select(max_fd+1, &rset, &wset, NULL, NULL);
  if (FD_ISSET(listenfd, &rset)) { //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;
      }
    FD_SET(connfd, &rfds);//加入到读的集合中,代表要监听某客户端的读(recv)事件
    //如果客户端没有回收,connfd比max_fd大,但是如果客户端回收了fd,那么数字会变小,因此需要判断当前的connfd是否大于max_fd  
    if (connfd > max_fd) max_fd = connfd;//重新设置最大的fd
    if (--nready == 0) continue;//只有一个客户端的连接
  }
  int i = 0;
  //注意0-标志输入,1-标志输出,2-错误,然后是listenfd,
  //然后每次客户端连接后,connfd会依次增大。回收后connfd会变小,但是无论如何都比listenfd大
  //因此这里检测的是所有客户端是否有客户端与服务器进行了通信
  for (i = listenfd+1;i <= max_fd;i ++) {
    if (FD_ISSET(i, &rset)) { //检测某个客户端是否已经准备好了数据或者关闭(正常关闭也会设置读数据)
    n = recv(i, buff, MAXLNE, 0);//
          if (n > 0) { //收到n个字节的数据
              buff[n] = '\0';
              printf("recv msg from client: %s\n", buff);
      设置可以往该客户端socket发送数据
      //有兴趣的朋友可以去为每个fd封装一个发送缓冲和接收缓冲,设置写标志后
      //然后再下面的else if (FD_ISSET(i, &wset))发送数据给客户端
      FD_SET(i, &wfds);
      send(i, buff, n, 0);//直接将接收到的数据发送给客户端
          } else if (n == 0) { //收到的字节为0,代表该客户端正常关闭了
      FD_CLR(i, &rfds);//清空该客户端的读事件
      //printf("disconnect\n");
              close(i);
          }
    if (--nready == 0) break;//没有客户端的读事件了
    } 
    else if (FD_ISSET(i, &wset)) {//某个客户端写事件已就绪
    // send(i, buff, n, 0);  //将前面接收到数据发送给客户端
    // FD_SET(i, &rfds);//
    }
  }
  } 
    close(listenfd);
    return 0;
}

说明

代码主要说明了select的基本用法,一般服务器使用select来实现多路复用的代码都跟其类似,不过在实际的开发中,一般select是和定时器事件结合起来的,因为在一般的网络中,包括定时器事件和网络I/O事件,在一次消息循环中需要同时处理定时器事件和网络I/O事件。,有兴趣的朋友可以参考live555的消息循环的相关代码。


相关文章
|
3月前
|
存储 运维 数据挖掘
服务器数据恢复—EqualLogic存储硬盘出现坏道的数据恢复案例
某品牌EqualLogic PS6100存储阵列上有一组由16块硬盘组建的raid5磁盘阵列。磁盘阵列上层划分多个大小不同的卷,存放虚拟机文件。 硬盘出现故障导致存储阵列不可用,需要恢复存储阵列中的数据。
|
3月前
|
存储 运维 Oracle
服务器数据恢复—存储硬盘指示灯亮黄灯,RAID5阵列崩溃的数据恢复案例
服务器存储数据恢复环境: 某单位一台某品牌DS5300存储,1个机头+4个扩展柜,50块的硬盘组建了两组RAID5阵列。一组raid5阵列有27块硬盘,存放Oracle数据库文件。存储系统上层一共划分了11个卷。 服务器存储故障: 存储设备上两个硬盘指示灯亮黄色。其中一组RAID5阵列崩溃,存储不可用,设备已经过保。
|
6月前
|
存储 Oracle 关系型数据库
服务器数据恢复—光纤存储上oracle数据库数据恢复案例
一台光纤服务器存储上有16块FC硬盘,上层部署了Oracle数据库。服务器存储前面板2个硬盘指示灯显示异常,存储映射到linux操作系统上的卷挂载不上,业务中断。 通过storage manager查看存储状态,发现逻辑卷状态失败。再查看物理磁盘状态,发现其中一块盘报告“警告”,硬盘指示灯显示异常的2块盘报告“失败”。 将当前存储的完整日志状态备份下来,解析备份出来的存储日志并获得了关于逻辑卷结构的部分信息。
|
7月前
|
数据挖掘
服务器数据恢复——服务器异常断电造成raid5阵列故障的数据恢复案例
某服务器上有一组由12块硬盘组建的raid5磁盘阵列。 机房供电不稳定导致机房中该服务器非正常断电,重启服务器后管理员发现服务器无法正常使用。 意外断电可能会导致服务器上的raid模块损坏。
|
4月前
|
Unix 应用服务中间件 索引
服务器数据恢复—LUN映射出错导致文件系统共享冲突的数据恢复案例
SUN光纤存储系统中有一组由6个硬盘组建的RAID6,划分为若干LUN,MAP到跑不同业务的服务器上,这些服务器上运行的是SOLARIS操作系统。 服务器不存在物理故障。由于公司业务变化,需要增加一台服务器跑新的应用。服务器管理员在原服务器在线的状态下,将其中一个lun映射到一台新服务器上。实际上,这个刚映射过去的卷已经map到了solaris生产系统上的某个lun上了。映射到新服务器后,服务器对这个卷进行初始化的操作,原solaris系统上的磁盘报错,重启服务器后这个卷已经无法挂载。 服务器管理员寻求sun原厂工程师的帮助。sun工程师检测后执行了fsck操作。执行完成后文件系统挂载成功。查
|
4月前
|
运维 监控 Java
Linux常用命令行大全:14个核心指令详解+实战案例
在服务器管理与开发运维领域,Linux 指令是构建技术能力体系的基石。无论是日常的系统监控、文件操作,还是复杂的服务部署与故障排查,熟练掌握指令的使用逻辑都是提升工作效率的核心前提。然而,对于初学者而言,Linux 指令体系往往呈现出“参数繁多易混淆”“组合使用门槛高”“实际场景适配难”等痛点——例如 ls 命令的 -l 与 -a 参数如何搭配查看隐藏文件详情,grep 与管道符结合时如何精准过滤日志内容,这些问题常常成为技术进阶的阻碍。
|
4月前
|
存储 数据挖掘 Linux
服务器数据恢复—重装系统导致OceanStor存储上的分区无法访问的数据恢复案例
服务器存储数据恢复环境: 华为OceanStor某型号存储+扩展盘柜,存储中的硬盘组建了raid5磁盘阵列,上层分配了1个lun。 linux操作系统,划分了两个分区,分区一通过lvm扩容,分区二为xfs文件系统。 服务器存储故障: 工作人员重装系统操作失误导致磁盘分区变化,分区二无法访问,数据丢失。
|
5月前
|
弹性计算 安全 Linux
阿里云服务器ECS安装宝塔Linux面板、安装网站(新手图文教程)
本教程详解如何在阿里云服务器上安装宝塔Linux面板,涵盖ECS服务器手动安装步骤,包括系统准备、远程连接、安装命令执行、端口开放及LNMP环境部署,手把手引导用户快速搭建网站环境。
|
5月前
|
存储 算法 数据挖掘
服务器数据恢复—昆腾存储StorNext文件系统数据恢复案例
一台昆腾存储设备中有一组raid5磁盘阵列。阵列上有两块硬盘先后离线,raid5磁盘阵列不可用。
|
4月前
|
存储 数据挖掘 Windows
服务器数据恢复—RAIDZ上层ZFS文件系统数据恢复案例
一台服务器有32块硬盘,采用Windows操作系统。 服务器在正常运行的时候突然变得不可用。没有异常断电、进水、异常操作、机房不稳定等外部因素。服务器管理员重启服务器,但是服务器无法进入系统。管理员联系北亚企安数据恢复工程师要求恢复服务器数据。