网络编程DAY4作业

本文展示了如何使用C语言构建并发服务器,包括基于多进程和多线程的两种方法。在多进程模型中,服务器接受新连接后创建子进程处理客户端请求,而主进程继续监听。在多线程模型中,服务器创建新的线程来处理每个客户端连接,主线程仅负责接收新的连接。两种模型都实现了基本的接收和回应客户端数据的功能。
摘要由CSDN通过智能技术生成

完成并发服务器的搭建

  • 1.多进程并发服务器
#include<string.h>
#include<stdio.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<unistd.h>
#include<signal.h>
#include<sys/wait.h>

#define ERR_MSG(msg) do{\
		fprintf(stderr, "line: %d\n",__LINE__);\
		perror(msg);\
}while(0)

#define IP "192.168.31.159"
#define tons 12345

int deal(int newfd,struct sockaddr_in cin);

void handler(int sig)
{
	while( waitpid(-1, NULL, WNOHANG) > 0);
}
int main(int argc, const char *argv[])
{
	//捕获17号信号,回收僵尸进程
	__sighandler_t s = signal(SIGCHLD, handler);
	if(SIG_ERR == s)
	{
		ERR_MSG("signal");
		return -1;
	}
	//创建流式套接字
	
	int sfd = socket(AF_INET , SOCK_STREAM , 0);
	if(sfd < 0)
	{
		ERR_MSG("socket");
		return -1;
	}

	//允许端口快速重复使用
	int reuse =1;
	if(setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) < 0)
	{
		ERR_MSG("setsockopt");
		return -1;
	}
	//绑定---,必须绑定    该真实结构体根据地址族指定:AF_INET:man 7 ip
	struct sockaddr_in sin;
	sin.sin_family = AF_INET;      //必须填充AF_INET
	sin.sin_port   = htons(tons); //端口号的网络字节序1024~49151
	sin.sin_addr.s_addr = inet_addr (IP);// IP,本机IP ifconfig

	if (bind(sfd, (struct sockaddr*)&sin, sizeof(sin)) < 0)
	{
		ERR_MSG("bind");
		return -1;
	}
	printf("bind success __%d__\n",__LINE__);
	//将套接字设置为被动监听状态
	if( listen (sfd, 128) < 0)
	{
		ERR_MSG("listen");
		return -1;
	}
	printf("listen success __%d__\n",__LINE__);


	struct sockaddr_in cin;                 //存储链接成功的客户端地址
	socklen_t addrlen = sizeof(cin);
	pid_t cpid;
	//从已完成连接的队列头中获取一个客户端信息,生成一个新的文件描述符
	while(1)
	{
		int newfd = accept(sfd ,(struct sockaddr*)&cin ,&addrlen);
		if(newfd < 0)
		{
			ERR_MSG("accept");	
		}
		printf("[%s : %d] newfd =%d 连接成功 __%d__\n",\
				inet_ntoa(cin.sin_addr), ntohs(cin.sin_port), newfd,__LINE__);

		cpid = fork();
		if(cpid > 0)
		{
			close(newfd);
		}else if(cpid < 0)
		{
			ERR_MSG("fork");
			return -1;
		}else if(cpid == 0)
		{
			close(sfd);
			if( deal(newfd,cin) < 0)
			{
				printf("deal default %d\n",__LINE__);
			}
			//关闭文件描述符	
			close(newfd);
			exit(0);   //运行到当前位置客户端退出,而不是while到accept
		}
	}
	close(sfd);

	return 0;
}


int deal(int newfd,struct sockaddr_in cin)
{
	while(1)
	{		
		char buf[128] = "";
		ssize_t res = 0;
		//清空,防止之前的数据干扰
		bzero(buf, sizeof(buf));
		//接收
		res = recv(newfd, buf, sizeof(buf), 0);
		if(res < 0)
		{
			ERR_MSG("recv");
			return -1;
		}else if(0 == res)
		{
			printf("[%s : %d] newfd =%d 客户端下线 __%d__\n",\
					inet_ntoa(cin.sin_addr), ntohs(cin.sin_port), newfd,__LINE__);	
			break;
		}
		printf("[%s : %d] newfd =%d : %s__%d__\n",\
				inet_ntoa(cin.sin_addr), ntohs(cin.sin_port), newfd,buf,__LINE__);
		//发送
		strcat(buf,"0.0\n");
		if( send(newfd, buf, sizeof(buf), 0) < 0)
		{
			ERR_MSG("send");
			return -1;		
		}
	}

	return 0;
}
  • 2.多线程并发服务器
#include<string.h>
#include<stdio.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<unistd.h>
#include<pthread.h>

#define ERR_MSG(msg) do{\
		fprintf(stderr, "line: %d\n",__LINE__);\
		perror(msg);\
}while(0)

#define IP "192.168.31.159"
#define tons 12345

struct cli_msg
{
	int newfd;
	struct sockaddr_in cin;
};

void* deal_cli_msg(void* arg);

int main(int argc, const char *argv[])
{
	//创建流式套接字
	int sfd = socket(AF_INET , SOCK_STREAM , 0);
	if(sfd < 0)
	{
		ERR_MSG("socket");
		return -1;
	}
	
	//允许端口快速重复使用
	int reuse =1;
	if(setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) < 0)
	{
		ERR_MSG("setsockopt");
		return -1;
	}
	//绑定---,必须绑定    该真实结构体根据地址族指定:AF_INET:man 7 ip
	struct sockaddr_in sin;
	sin.sin_family = AF_INET;      //必须填充AF_INET
	sin.sin_port   = htons(tons); //端口号的网络字节序1024~49151
	sin.sin_addr.s_addr = inet_addr (IP);// IP,本机IP ifconfig

	if (bind(sfd, (struct sockaddr*)&sin, sizeof(sin)) < 0)
	{
		ERR_MSG("bind");
		return -1;
	}
	printf("bind success __%d__\n",__LINE__);
	//将套接字设置为被动监听状态
	if( listen (sfd, 128) < 0)
	{
		ERR_MSG("listen");
		return -1;
	}
	printf("listen success __%d__\n",__LINE__);


	struct sockaddr_in cin;                 //存储链接成功的客户端地址
	socklen_t addrlen = sizeof(cin);
	int newfd = -1;
	pthread_t tid;
	struct cli_msg info;

	//从已完成连接的队列头中获取一个客户端信息,生成一个新的文件描述符
	while(1)
	{
		//主线程只负责连接
		//accept函数阻塞之前会预选一个没有使用过的文件描述符作为newfd
		//当解除阻塞的时候,如果预选的newfd没有被使用过,则直接返回预选的文件描述符newfd
		//如果预选的newfd被使用过了,则需要重现遍历文件描述符表,拿到一个没有被使用过的文件描述符作为newfd
		newfd = accept(sfd ,(struct sockaddr*)&cin ,&addrlen);
		if(newfd < 0)
		{
			ERR_MSG("accept");	
			return -1;
		}
		printf("[%s : %d] newfd =%d 连接成功 __%d__\n",\
				inet_ntoa(cin.sin_addr), ntohs(cin.sin_port), newfd,__LINE__);

		info.newfd = newfd;
		info.cin = cin;
		
		//能运行到当前位置,则代表有客户端连接成功
		//需要创建一个分支线程,用于与客户端交互使用
		if(pthread_create(&tid, NULL, deal_cli_msg, (void*)&info) != 0)
		{
			fprintf(stderr,"line :%d pthread_create failed\n",__LINE__);
			return -1;
		}
		pthread_detach(tid);        //分离线程
	}

	close(sfd);
	return 0;
}

void* deal_cli_msg(void* arg)
{
	//必须另存
	int newfd = (*(struct cli_msg*)arg).newfd;
	struct sockaddr_in cin = (*(struct cli_msg*)arg).cin;

	char buf[128] = "";
	ssize_t res = 0;
	while(1)
	{
		//清空,防止之前的数据干扰
		bzero(buf, sizeof(buf));
		//接收
		res = recv(newfd, buf, sizeof(buf), 0);
		if(res < 0)
		{
			ERR_MSG("recv");
			break;
		}else if(0 == res)
		{
			printf("[%s : %d] newfd =%d 客户端下线 __%d__\n",\
					inet_ntoa(cin.sin_addr), ntohs(cin.sin_port), newfd,__LINE__);	
			break;
		}
		printf("[%s : %d] newfd =%d : %s__%d__\n",\
				inet_ntoa(cin.sin_addr), ntohs(cin.sin_port), newfd,buf,__LINE__);
		//发送
		strcat(buf,"0.0\n");
		if( send(newfd, buf, sizeof(buf), 0) < 0)
		{
			ERR_MSG("send");
			break;	
		}
		printf("发送成功\n");
	}
	//关闭文件描述符	
	close(newfd);
	pthread_exit(NULL);

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值