多路复用——select

        在之前我们说过TCP的模型,用户可以通过连接服务器从而通过服务器去加载服务端对应功能的文件来达到通信,但是我们知道,一个用户加载文件需要一个进程,在Linux中,一个进程就要占用4G的虚拟内存,如果在并发量高的情况下,很明显服务器是不堪重负的。下面来讲解决方案即多路复用技术。

        多路复用,举个例子就好比把多条狭窄的乡间小路修成一条双向16车道的快速路,即在这条大路上去进行网络通信。多路复用技术的核心就在于在单个网络信道上传输多个网络信号,能能大幅提高通信效率,并且能够很大程度上降低服务器的负担。

下面我们用TCP作为通信模型来看多路复用的第一种 I/O 机制,即select:

服务端如下:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/select.h>

//tcp server 只有server端能用多路复用
int main(int argc, char *argv[])
{
    //创建socket
    int svr_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (svr_fd < 0)
     {
        perror("socket");
        return -1;
    }
    //准备通信地址
    struct sockaddr_in addr;
    addr.sin_family = AF_INET;
    addr.sin_port = htons(8888);
    addr.sin_addr.s_addr = inet_addr("127.0.0.1");
    socklen_t addrlen=sizeof(addr);
    //绑定地址
    if (bind(svr_fd, (struct sockaddr*)&addr, addrlen) < 0)
    {
        perror("bind");
        return -1;
    }
    //监听连接
    if (listen(svr_fd, 10))
    {
        perror("listen");
        return -1;
    }
    //定义一个文件描述符集合并清空
 	fd_set reads;
	FD_ZERO(&reads);
	//把需要等待的socket描述符添加到集合中
	FD_SET(svr_fd,&reads);
	//定义超时时间
	struct timeval timeout={5,500};
	//记录最大的socket描述符
	int max_fd=svr_fd;//当前socket描述符就这一个,所以也是最大值

	char buf[4096];
	size_t buf_size=sizeof(buf);

	while(1)
	{
		//若有多个新的连接时,集合会发生变化,而把之前的没发生变化的文件描述符给覆盖了
		fd_set reads_copy=reads;//备份集合
		int ret=select(max_fd+1,&reads_copy,NULL,NULL,&timeout);
		if(ret>0)
		{
			//一开始,先测试网络等待的socket描述符
			if(FD_ISSET(svr_fd,&reads_copy))
			{
				//调用accpet连接客户端
				int cli_fd=accept(svr_fd,(struct sockaddr*)&addr,&addrlen);
				if(cli_fd<0)
				{
					perror("accept");
				}
				else
				{
					//把客户端的socket描述符添加到监控集合中
					FD_SET(cli_fd,&reads);
					if(cli_fd>max_fd)
					{
						max_fd=cli_fd;
					}
				}
			}
			else
			{
				//测试其他socket描述符是否发生变化
				for(int fd=3;fd<=max_fd;fd++)
				{
					if(FD_ISSET(fd,&reads_copy)&&fd!=svr_fd)
					{
						int ret=recv(fd,buf,buf_size,0);
						if(ret<=0)
						{
							FD_CLR(fd,&reads);
							printf("客户端%d退出\n",fd);
							continue;
						}
						printf("recv:%s bits:%d\n",buf,ret);
						strcat(buf,":return");
						ret=send(fd,buf,strlen(buf)+1,0);
						if(ret<=0 || 0==strcmp("quit",buf))
						{
							FD_CLR(fd,&reads);
							printf("客户端%d退出\n",fd);
							continue;
						}
					}
				}
			}
		}
	}

	return 0;
}

客户端如下:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/select.h>


typedef struct sockaddr *SP;

int main(int argc,const char* argv[])
{
	//创建socket
	int cli_fd=socket(AF_INET,SOCK_STREAM,0);
	if(cli_fd<0)
	{
		perror("socket");
		return -1;
	}
	//准备通信地址
	struct sockaddr_in addr={};
	addr.sin_family=AF_INET;
	addr.sin_port=htons(8888);
	addr.sin_addr.s_addr=inet_addr("127.0.0.1");
	socklen_t addrlen=sizeof(addr);
	//连接服务器
	if(connect(cli_fd,(SP)&addr,addrlen))
	{
		perror("connect");
		return -1;
	}

	char buf[4096];
	size_t buf_size=sizeof(buf);
	while(1)
	{
		//发送请求
		printf(">>>>>");
		scanf("%s",buf);
		int ret=send(cli_fd,buf,strlen(buf)+1,0);
		//ret=write(cli_fd,buf,strlen(buf)+1);
		if(ret<=0)
		{
			printf("服务器正在升级,请稍后重试\n");
			break;
		}
		if(0==strcmp("quit",buf))
		{
			printf("通信结束\n");
			break;
		}
		//接收请求
		//int ret=read(cli_fd,buf,buf_size);
		ret=recv(cli_fd,buf,buf_size,0);
		if(ret<=0)
		{
			printf("服务器正在维护,请稍候重试\n");
			break;
		}
		printf("read:%s bits:%d\n",buf,ret);
	}
		
	return 0;
}

over

select函数是Linux C编程中的一种多路复用机制,用于监视并等待多个文件描述符的属性发生变化。它可以同时监视多个文件描述符的可读、可写和异常状态。select函数的原型如下: ```c #include <sys/select.h> int select(int nfds, fd_set* readfds, fd_set* writefds, fd_set* exceptfds, struct timeval* timeout); ``` 其中,nfds是待监视的文件描述符的最大值加1,readfds、writefds和exceptfds分别是用于监视可读、可写和异常状态的文件描述符集合。timeout参数指定等待时间,当超过指定时间后,select函数会返回。当select函数返回后,可以通过遍历fd_set集合来确定哪些文件描述符已经就绪。 使用select函数可以实现多个文件描述符的并发处理,提高程序的效率和响应速度。 #### 引用[.reference_title] - *1* [IO多路复用之——select详解](https://blog.csdn.net/weixin_60954394/article/details/127062613)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [Linux下C语言多路复用——select函数](https://blog.csdn.net/qq_45097019/article/details/105166595)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值