select创建一个简单的服务器

一:原理

select实现了一个多路复用的概念,那么何为多路复用,简单理解就是可以把多个分散的IO事件(读,写,error)集中到一个地方处理,就网络而言,通过select,可以将多个客服端的socket加入一个集合,然后轮询这个集合查找你感兴趣的事件

二:用处

就网络而言,在服务器这层,当有多个客户端同时连接进服务器时,不必开多线程去接受连接,也不会出现单线程阻塞等待的情况

三:使用

第一步:创建一个监听socket
第二步:创建一个fd集合
第三步:将监听socket加入集合
第四步:监听这个集合,如果发现是监听socket,则往里面该添加客户端socket,如果不是那么就是客户端socket,则进行业务操作

#include <iostream>
#include <netinet/in.h>
#include <sys/types.h>
#include <sys/select.h>
#include <sys/socket.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <cstring>
#define PORT 8080
#define CLIENTCOUNT 10
int main(){

    //第一步
    int socket_fd = socket(AF_INET, SOCK_STREAM, 0);
    sockaddr_in addr;
    memset(&addr, 0, sizeof(sockaddr_in));

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

    bind(socket_fd, (sockaddr *)&addr, sizeof(sockaddr));

    if (listen(socket_fd, 5) == 0)
    {
        std::cout << "listen create succ! " << socket_fd << std::endl;
    }
    else
    {
        std::cout << "listen create fail!" << std::endl;
    }
	//第二步
    fd_set read_set;
    FD_ZERO(&read_set);
    //第三步
    FD_SET(socket_fd, &read_set);

    int maxfd = socket_fd;
    while (true)
    {
    	//注意这个tmp_set 一定要有,因为在监听过程中 集合里面的值是在改变的,所以使用一个局部的值
        fd_set tmp_set;
        FD_ZERO(&tmp_set);
        tmp_set = read_set;
		
		//第四步,这一步 会改变 tmp_set 
        int read_fd = select(maxfd + 1, &tmp_set, nullptr, nullptr, nullptr);
        std::cout << "read_fd:" << read_fd << std::endl;
        if (read_fd == -1)
        {
            std::cout << "select err" << std::endl;
            break;
        }

        for (int i = 0; i < maxfd + 1; i++)
        {

            if (FD_ISSET(i, &tmp_set))
            {

                if (i == socket_fd)
                { //服务器文件描述符

                    sockaddr_in client_addr;
                    memset(&client_addr, 0, sizeof(sockaddr));
                    socklen_t addrlen = sizeof(sockaddr_in);
                    int client_fd = accept(i, (sockaddr *)&client_addr, &addrlen);
                    //std::cout << "client fd:" << client_fd << "," << i << std::endl;
                    std::cout << "client:" << inet_ntoa(client_addr.sin_addr)<< std::endl;
                    FD_SET(client_fd, &read_set);
                    if(maxfd < client_fd)
                        maxfd = client_fd;
                    std::string tips = "connect succ!!!";
                    send(client_fd, tips.c_str(), tips.length(), 0);
                    
                    // fflush(client_fd);
                    // break;
                }
                else
                {

                    char buff[1024] = {0};

                    int ret = read(i, buff, 1024);

                    if (ret == 0)
                    {
                        std::cout << "client disconnet" << std::endl;
                        maxfd --;
                        close(i);
                        FD_CLR(i, &read_set);

                        
                    }
                    else if (ret == -1)
                    {
                        std::cout << "recv err" << buff << std::endl;
                        
                    }
                    else
                    {

                        std::cout << "rec :"<<i<<"," << buff << std::endl;
                        write(i,buff,ret);
                        
                    }
                }
            }
        }
    }


    close(socket_fd);

    std::cout << "server close" << std::endl;
    return 0;
}

四:不足

轮询的机制导致了select会遍历所有集合内的IO,这就是它低效的主要原因,举个例子:假设一个服务器有100个客户端连接,那么在服务端就会有100个socket对应着这些客户端,如果此时有一个客户端发送了数据到服务端,那么为了接收这一次的数据到应用层,select需要遍历这100个socket状态,很明显有99个遍历过程是浪费的

五:解决不足

select 本身是一个多路复用的实现,在IO处理较多的情况下,比之普通的单线程要高效多了,说它低效是因为有一个更高效的实现存在,那就是epoll,epoll在select的基础上又增加了一个集合用来管理已经触发了事件的IO。

六:epoll创建一个简单的服务器

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
下面是一个简单的C语言程序,用于创建TCP服务器并监听客户端连接请求: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/socket.h> #include <arpa/inet.h> #include <unistd.h> #define MAX_CLIENTS 10 #define BUFFER_SIZE 1024 int main(int argc, char *argv[]) { int server_fd, client_fds[MAX_CLIENTS], max_sd, activity, new_socket, valread; struct sockaddr_in server_addr; char buffer[BUFFER_SIZE]; // 创建服务器套接字 if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) { perror("server socket failed"); exit(EXIT_FAILURE); } // 设置服务器套接字选项 if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR, &(int){1}, sizeof(int)) < 0) { perror("setsockopt failed"); exit(EXIT_FAILURE); } // 绑定服务器套接字 server_addr.sin_family = AF_INET; server_addr.sin_addr.s_addr = INADDR_ANY; server_addr.sin_port = htons(8888); if (bind(server_fd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) { perror("bind failed"); exit(EXIT_FAILURE); } // 监听客户端连接请求 if (listen(server_fd, MAX_CLIENTS) < 0) { perror("listen failed"); exit(EXIT_FAILURE); } printf("Waiting for connections...\n"); // 处理客户端连接请求 while (1) { fd_set readfds; // 初始化文件描述符集 FD_ZERO(&readfds); // 添加服务器套接字到文件描述符集 FD_SET(server_fd, &readfds); max_sd = server_fd; // 添加客户端套接字到文件描述符集 for (int i = 0; i < MAX_CLIENTS; i++) { int sd = client_fds[i]; if (sd > 0) FD_SET(sd, &readfds); if (sd > max_sd) max_sd = sd; } // 等待读取文件描述符集中的任何套接字 activity = select(max_sd + 1, &readfds, NULL, NULL, NULL); if ((activity < 0) && (errno != EINTR)) { printf("select error\n"); } // 如果有新的客户端连接请求 if (FD_ISSET(server_fd, &readfds)) { if ((new_socket = accept(server_fd, (struct sockaddr *)&server_addr, (socklen_t*)&server_addr)) < 0) { perror("accept failed"); exit(EXIT_FAILURE); } printf("New connection, socket fd is %d, address is %s, port is %d\n", new_socket, inet_ntoa(server_addr.sin_addr), ntohs(server_addr.sin_port)); // 将新的客户端套接字添加到客户端套接字数组中 for (int i = 0; i < MAX_CLIENTS; i++) { if (client_fds[i] == 0) { client_fds[i] = new_socket; break; } } } // 处理客户端套接字的数据 for (int i = 0; i < MAX_CLIENTS; i++) { int sd = client_fds[i]; if (FD_ISSET(sd, &readfds)) { if ((valread = read(sd, buffer, BUFFER_SIZE)) == 0) { // 客户端关闭连接 printf("Client disconnected, socket fd is %d\n", sd); close(sd); client_fds[i] = 0; } else { // 处理客户端发送的数据 buffer[valread] = '\0'; printf("Received data from client, socket fd is %d: %s\n", sd, buffer); send(sd, buffer, strlen(buffer), 0); } } } } return 0; } ``` 这个程序创建一个TCP服务器套接字,然后绑定到端口8888,并开始监听客户端连接请求。当有新的客户端连接请求时,它会接受连接并将客户端套接字添加到客户端套接字数组中。然后它使用select函数等待读取客户端套接字中的数据,一旦有数据可读,程序就会读取数据并将其发送回客户端。如果客户端关闭连接,它将从客户端套接字数组中删除该套接字。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

dai1396734

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值