linux——TCP多线程并发服务器

多线程服务器 = 可以同时处理多个客户端

旧版:一次只能接一个客户,客户不走,别人连不进来。

新版:来一个客户,创建一个线程专门服务,同时支持 N 个客户端!

主函数加了一个while(1)循环

复制代码
pthread_t tid;

	int newfd = -1;
	struct sockaddr_in cin;
	socklen_t addrlen = sizeof(cin);
	while(1)
	{
		newfd = accept(fd,(struct sockaddr *)&cin,&addrlen);
		if(newfd < 0)
		{
			perror("accept");
			exit(1);
		}

		char ipv4_addr[16];
		if(!inet_ntop(AF_INET,(void *)&cin.sin_addr,ipv4_addr,sizeof(cin)))
		{
			perror("inet_ntop");
			exit(1);
		}
		printf("Client:(%s,%d) is connect\n",ipv4_addr,ntohs(cin.sin_port));

		pthread_create(&tid,NULL,client_data_handle,(void *)&newfd);
	}

就是无线循环accept函数

accept 接到一个客户 → 立刻创建一个线程去服务它 → 主线程马上回去继续等下一个客户!

创建线程

复制代码
pthread_create(&tid, NULL, client_data_handle, (void *)&newfd);

让这个线程去执行 client_data_handle 函数专门服务当前这个客户端

主线程马上回去继续 accept 等下一个客户!

线程函数用来接收数据代码

复制代码
void* client_data_handle(void* arg)
{
	int newfd = *(int *)arg;

	char buf[BUFSIZE];
	int ret = -1;
	printf("handle thread :newfd = %d\n",newfd);

	while(1)
	{
		do
		{
			bzero(buf,BUFSIZE);
			ret = read(newfd,buf,BUFSIZE-1);
		}while(ret < 1);
		if(ret < 0)
		{
			exit(1);
		}
		if(!ret)
		{
			break;
		}
		printf("receive data:%s\n",buf);

		if(!strncasecmp(buf,QUIT_STR,strlen(QUIT_STR)))
		{
			printf("Client is exiting!\n");
			break;
		}
	}

	close(newfd);
	return NULL;
}

每个客户端,都由一个独立的线程服务!

同时也把客户端代码优化一下,以前的客户端代码,IP地址和端口号是写死的,我们改成命令行传参

复制代码
int main(int argc, char **argv)
{
    port = atoi(argv[2]);
    sin.sin_addr.s_addr = inet_addr(argv[1]);
}

这样客户端更灵活,相连接不同的服务器,不用总是修改代码,想连谁在终端输入对应的IP地址和端口号就行

服务器端

复制代码
#include<stdio.h>
#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>
#include <unistd.h>
#include<stdlib.h>
#include <strings.h>
#include <arpa/inet.h>
#include<string.h>
#include <pthread.h>

#define QUIT_STR "QUIT"
#define BUFSIZE 1024
#define BACKLOG 5
#define SERV_IP 5001
#define SERV_IP_ADDR "192.168.88.129"

void* client_data_handle(void* arg);
int main()
{
	int fd = -1;

	struct sockaddr_in sin;
	//1.socket
	fd = socket(AF_INET,SOCK_STREAM,0);
	if(fd<0)
	{
		perror("socket");
		exit(1);
	}

	bzero(&sin,sizeof(sin));

	sin.sin_family = AF_INET;
	sin.sin_port = htons(SERV_IP);
	//sin.sin_addr.s_addr = inet_addr(SERV_IP_ADDR);  
	sin.sin_addr.s_addr = INADDR_ANY;
	/*if(inet_pion(AF_INET,SERV_IP_ADDR,(void *)sin.sin_addr.s_addr) != 1)
	  {
	  perror("inet_pton");
	  exit(1);
	  }
	  */
	//2.bind
	if(bind(fd,(struct sockaddr *)&sin,sizeof(sin)) <0)
	{
		perror("bind");
		exit(0);
	}
	//3.listen
	if(listen(fd,BACKLOG) < 0)
	{
		perror("listen");
		exit(1);
	}

	//4.accept
	/*int newfd = -1;
	  newfd = accept(fd,NULL,NULL);
	  if(newfd < 0)
	  {
	  perror("accept");
	  exit(1);
	  }
	  */

	pthread_t tid;

	int newfd = -1;
	struct sockaddr_in cin;
	socklen_t addrlen = sizeof(cin);
	while(1)
	{
		newfd = accept(fd,(struct sockaddr *)&cin,&addrlen);
		if(newfd < 0)
		{
			perror("accept");
			exit(1);
		}

		char ipv4_addr[16];
		if(!inet_ntop(AF_INET,(void *)&cin.sin_addr,ipv4_addr,sizeof(cin)))
		{
			perror("inet_ntop");
			exit(1);
		}
		printf("Client:(%s,%d) is connect\n",ipv4_addr,ntohs(cin.sin_port));

		pthread_create(&tid,NULL,client_data_handle,(void *)&newfd);
	}
	close(fd);

	return 0;
}

void* client_data_handle(void* arg)
{
	int newfd = *(int *)arg;

	char buf[BUFSIZE];
	int ret = -1;
	printf("handle thread :newfd = %d\n",newfd);

	while(1)
	{
		do
		{
			bzero(buf,BUFSIZE);
			ret = read(newfd,buf,BUFSIZE-1);
		}while(ret < 1);
		if(ret < 0)
		{
			exit(1);
		}
		if(!ret)
		{
			break;
		}
		printf("receive data:%s\n",buf);

		if(!strncasecmp(buf,QUIT_STR,strlen(QUIT_STR)))
		{
			printf("Client is exiting!\n");
			break;
		}
	}

	close(newfd);
	return NULL;
}

客户端

复制代码
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<strings.h>
#include<unistd.h>
#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>
#include <arpa/inet.h>
//  ./client 192.168.88.129   5001



#define SERV_PORT 5001
#define SERV_IP_ADDR "192.168.88.129"
#define BUFSIZE 1024
#define QUIT_STR "QUIT"

int main(int argc,char **argv)
{
	int fd = -1;
	if(argc != 3)
	{
		exit(1);
	}
	int port = -1;
	port = atoi(argv[2]);

	struct sockaddr_in sin;
	fd = socket(AF_INET,SOCK_STREAM,0);
	if(fd < 0)
	{
		perror("socket");
		exit(1);
	}

	bzero(&sin,sizeof(sin));

	sin.sin_family = AF_INET;
	sin.sin_port = htons(port);
	sin.sin_addr.s_addr = inet_addr(argv[1]);

	if(connect(fd,(struct sockaddr*)&sin,sizeof(sin)) < 0)
	{
		perror("connect");
		exit(1);
	}

	char buf[BUFSIZE];
	while(1)
	{
		bzero(buf,BUFSIZE);

		if(fgets(buf,BUFSIZE-1,stdin) == NULL)
		{
			continue;
		}
		write(fd,buf,strlen(buf));

		if(!strncasecmp(buf,QUIT_STR,strlen(QUIT_STR)))
		{
			break;
		}
	}

	return 0;
}

成功实现了

相关推荐
csdn_aspnet几秒前
Modbus TCP C# 客户端程序
服务器·网络·tcp/ip·c#
辣椒思密达1 分钟前
住宅IP与机房IP的区别及技术选型指南
网络·网络协议·tcp/ip
沐风___3 分钟前
App 上架之后:如何看数据、获取用户与持续迭代产品
服务器·前端·数据库
小灰灰搞电子5 分钟前
Rust 实现异步ModbusTCP主机源码分享
服务器·网络·modbustcp·rust
apcipot_rain16 分钟前
计科八股20260529——连接协议连接线程池、模块拆解模块通信、WebSocket
运维·服务器·网络·八股
AAA大运重卡何师傅(专跑国道)41 分钟前
力扣hot100
服务器·前端·数据库
TechWayfarer42 分钟前
IP精准定位服务在快递网点规划中的应用:如何用客户位置数据辅助选址
大数据·网络·python·tcp/ip·交通物流
剑神一笑1 小时前
Linux lsof 命令深度解析:从文件描述符到进程追踪
linux·运维·php
qq3621967051 小时前
facebook是什么意思?新手从零到精通完全指南
运维·服务器·facebook
andlbds1 小时前
解决Ubuntu20.04进入系统卡死在厂商Logo界面问题
linux·ubuntu