网络编程day4

多进程实现TCP并发服务器

结果图:

代码

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

#define SER_PORT 8888
#define SER_IP  "192.168.124.108"     // "0.0.0.0"

int deal_cli_msg(int newfd,struct sockaddr_in c);

// 当子进程退出后,父进程回收到17) SIGCHLD 
// 修改17号信号的默认处理函数,在17号信号的默认处理函数中给子进程收尸
void signal_handle(int signum)
{
	
		// 阻塞函数,阻塞等待任意子进程退出,若没有子进程,则wait函数运行失败,返回-1,
		// 若回收成功,则返回回收到的子进程的pid号
		//wait(NULL);
		
		//waitpid(-1,NULL,0);   //等待指定的进程退出,填-1,代表任意子进程,然后填0 和wait一样阻塞
	pid_t wpid = waitpid(-1,NULL,WNOHANG);      //非阻塞方式回收任意子进程
	printf("wpid=%d\n",wpid);								// 此时有子进程,但是子进程没有退出,返回0,不阻塞
									// 若此时有子进程,且子进程已经退出了,不阻塞,
									//  返回子进程的pid
									// 此时没有子进程,函数运行失败,返回-1
	
}




int main(int argc,char **argv)
{
	// 给17号信号SIGCHLD信号注册新的处理函数
	if(signal(SIGCHLD,signal_handle) == SIG_ERR)
	{
		fprintf(stderr,"__%d__",__LINE__);
		perror("signal");
		return -1;
	}
	printf("SIGCHLD change success\n");
	
	// 创建流式套接字
	int sfd = socket(AF_INET,SOCK_STREAM,0);
	if(sfd <0){
		fprintf(stderr,"__%d__",__LINE__);
		perror("socket");
		return -1;
	}
	printf("sokect success __%d__\n",__LINE__);
	
	// 允许端口快速重用
	int reuse = 1;
	if(setsockopt (sfd, SOL_SOCKET,SO_REUSEADDR, &reuse,sizeof(reuse))<0)
	{
        fprintf(stderr, "__%d__ ", __LINE__);
        perror("setsockopt");
        return -1;
    }                                                                              
    printf("reuseaddr success __%d__\n", __LINE__);
	
	// 若从命令行传入端口号,则使用传入的端口号,没有传入,就使用默认端口号
	int port = argc>=2 ? atoi(argv[1]):SER_PORT;
	
	// 填充服务器的地址信息结构体,给bind函数使用
	// 真实的地址信息结构体
	struct sockaddr_in sin;
	sin.sin_family  = AF_INET;  //必须填AF_INET
	sin.sin_port   = htons(port);    //端口号的网络字节序
	sin.sin_addr.s_addr   = inet_addr(SER_IP);   // 本机IP的网络字节序
									// 127.0.0.1  本地环回IP,只能做本机通信
									// 0.0.0.0  代表运行环境中所有可用IP

	// 绑定服务器自身的地址信息
	
	if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin)) <0){
		fprintf(stderr,"__%d__",__LINE__);
		perror("bind");
		return -1;
	}
	printf("bind success __%d__\n",__LINE__);
	
	// 将套接字设置为被动监听状态
	if(listen(sfd,128)<0)            // 128 未完全连接的队列里面的客户机数量
	{
		fprintf(stderr,"__%d__",__LINE__);
		perror("listen");
		return -1;
	}
	printf("listen success __%d__\n",__LINE__);
	
	// 阻塞等待客户端连接成功,从已完成连接的队列头中获取一个客户端信息
	// 生成一个新的文件描述符,这个新的文件描述符才是与客户端通信的文件描述符
	struct sockaddr_in cin;  //  存储获取到的客户端地址信息	
	socklen_t addrlen = sizeof(cin);
	
	//accept(sfd,NULL,NULL);  // 不存储从已完成连接队列中获取到的客户端信息
	
		

	int newfd = -1;
	pid_t pid= 0;
	while(1)
	{
		// 父进程代码
		newfd = accept(sfd,(struct sockaddr*)&cin,&addrlen);
		if(newfd<0)
		{
			fprintf(stderr,"__%d__",__LINE__);
			perror("accept");
			return -1;
		}
		printf("accept success ,newfd = %d,__%d__\n",newfd,__LINE__);

	
		pid =fork();
		if(0 == pid){
			 close(sfd);    // sfd 对于子进程没有用
			 deal_cli_msg(newfd,cin);
			 break;  // exit(0)
		}
		close(newfd);
		
		
	}
	// 关闭
	close(sfd);     // 监听newfd
	return 0;
}	


// 子进程处理客户端交互
int deal_cli_msg(int newfd,struct sockaddr_in cin)
{
	char buf[128]="";
	ssize_t res =0;
	
	while(1){
		bzero(buf,sizeof(buf));  // memset(buf,0,sizeof(buf));
		//收   //会再这里等待。
		
		res = recv(newfd,buf,sizeof(buf),0);   // 返回收到的字符
		if(res<0){
			fprintf(stderr,"__%d__",__LINE__);
			perror("recv");
			return -1;
		}
		else if(0 == res){      // 说明客户端已经结束了
			printf("client offline [%s:%d] __%d__\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),__LINE__);
			break;
		}
		// 打印收到的客户端的ip和端口
		printf("client [%s:%d]newfd %d:%s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),newfd,buf);

	
		
		//发
		strcat(buf,"*_*");
		if(send(newfd, buf, sizeof(buf), 0)<0){
			fprintf(stderr,"__%d__",__LINE__);
			perror("send");
			return -1;
		}	
	}
	return 0;
	
}

 多线程实现TCP并发服务器

结果图:

代码:

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <string.h>
#include <unistd.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <pthread.h>

#define SER_PORT 8888
#define SER_IP  "192.168.124.108"     // "0.0.0.0"

//结构体 :需要传入到线程中内容
struct Climsg
{
	int newfd;
	struct sockaddr_in cin;
};




void* deal_cli_msg(void* arg);


int main(int argc,char **argv)
{
	
	
	// 创建流式套接字
	int sfd = socket(AF_INET,SOCK_STREAM,0);
	if(sfd <0){
		fprintf(stderr,"__%d__",__LINE__);
		perror("socket");
		return -1;
	}
	printf("sokect success __%d__\n",__LINE__);
	
	// 允许端口快速重用
	int reuse = 1;
	if(setsockopt (sfd, SOL_SOCKET,SO_REUSEADDR, &reuse,sizeof(reuse))<0)
	{
        fprintf(stderr, "__%d__ ", __LINE__);
        perror("setsockopt");
        return -1;
    }                                                                              
    printf("reuseaddr success __%d__\n", __LINE__);
	
	// 若从命令行传入端口号,则使用传入的端口号,没有传入,就使用默认端口号
	int port = argc>=2 ? atoi(argv[1]):SER_PORT;
	
	// 填充服务器的地址信息结构体,给bind函数使用
	// 真实的地址信息结构体
	struct sockaddr_in sin;
	sin.sin_family  = AF_INET;  //必须填AF_INET
	sin.sin_port   = htons(port);    //端口号的网络字节序
	sin.sin_addr.s_addr   = inet_addr(SER_IP);   // 本机IP的网络字节序
									// 127.0.0.1  本地环回IP,只能做本机通信
									// 0.0.0.0  代表运行环境中所有可用IP

	// 绑定服务器自身的地址信息
	
	if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin)) <0){
		fprintf(stderr,"__%d__",__LINE__);
		perror("bind");
		return -1;
	}
	printf("bind success __%d__\n",__LINE__);
	
	// 将套接字设置为被动监听状态
	if(listen(sfd,128)<0)            // 128 未完全连接的队列里面的客户机数量
	{
		fprintf(stderr,"__%d__",__LINE__);
		perror("listen");
		return -1;
	}
	printf("listen success __%d__\n",__LINE__);
	
	// 阻塞等待客户端连接成功,从已完成连接的队列头中获取一个客户端信息
	// 生成一个新的文件描述符,这个新的文件描述符才是与客户端通信的文件描述符
	struct sockaddr_in cin;  //  存储获取到的客户端地址信息	
	socklen_t addrlen = sizeof(cin);
	
	//accept(sfd,NULL,NULL);  // 不存储从已完成连接队列中获取到的客户端信息
	
		
	pthread_t tid;  // 存储线程tid
	int newfd = -1;
	struct Climsg pcli;
	
	while(1)
	{
		// 主线程代码
		newfd = accept(sfd,(struct sockaddr*)&cin,&addrlen);
		if(newfd<0)
		{
			fprintf(stderr,"__%d__",__LINE__);
			perror("accept");
			return -1;
		}
		printf("accept success ,newfd = %d,__%d__\n",newfd,__LINE__);

		// 能运行到这个位置吗,代表有客户端连接成功了
		// 需要创建一个分支线程,专门用于与客户端及进行交互(reve/send)
		pcli.newfd = newfd;
		pcli.cin = cin;
		
		pthread_create(&tid,NULL,deal_cli_msg,(void*)&pcli);   // 创建线程
		pthread_detach(tid);    //分离线程
		
		
	}
	// 关闭
	close(sfd);     // 监听newfd
	return 0;
}	


// 子进程处理客户端交互
void* deal_cli_msg(void* arg)
{
	int newfd = ((struct Climsg*)arg)->newfd;
	struct sockaddr_in cin = ((struct Climsg*)arg)->cin;
	
	char buf[128]="";
	ssize_t res =0;
	
	while(1){
		bzero(buf,sizeof(buf));  // memset(buf,0,sizeof(buf));
		//收   //会再这里等待。
		
		res = recv(newfd,buf,sizeof(buf),0);   // 返回收到的字符
		if(res<0){
			fprintf(stderr,"__%d__",__LINE__);
			perror("recv");
			break;
		}
		else if(0 == res){      // 说明客户端已经结束了
			printf("client offline [%s:%d] __%d__\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),__LINE__);
			break;
		}
		// 打印收到的客户端的ip和端口
		printf("client [%s:%d]newfd %d:%s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),newfd,buf);

	
		
		//发
		strcat(buf,"*_*");
		if(send(newfd, buf, sizeof(buf), 0)<0){
			fprintf(stderr,"__%d__",__LINE__);
			perror("send");
			break;
		}	
		printf("send success\n");
		
	}
	// 关闭
	close(newfd);
	pthread_exit(NULL);
}

select实现TCP并发服务器

结果图:

代码:

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

#define SER_PORT 8888
#define SER_IP  "192.168.124.108"     // "0.0.0.0"

// 函数声明
int deal_cliRecvSend(int i,struct sockaddr_in savecin[],fd_set *preadfds,int *pmaxfd);  //客户端交互函数
int deal_keyboard_msg(fd_set readfds);                                                    // 键盘触发
int deal_cliConnect(int sfd,struct sockaddr_in psavecin[],fd_set *peadfds,int *pmaxfd);  // 客户端连接函数

int main(int argc,char **argv)
{
	// 创建流式套接字
	int sfd = socket(AF_INET,SOCK_STREAM,0);
	if(sfd <0){
		fprintf(stderr,"__%d__",__LINE__);
		perror("socket");
		return -1;
	}
	printf("sokect success __%d__\n",__LINE__);
	
	// 允许端口快速被复用
	int reuse = 1;
	if(setsockopt (sfd, SOL_SOCKET,SO_REUSEADDR, &reuse,sizeof(reuse))<0)
	{
        fprintf(stderr, "__%d__ ", __LINE__);
        perror("setsockopt");
        return -1;
    }                                                                              
    printf("reuseaddr success __%d__\n", __LINE__);
	
	// 若从命令行传入端口号,则使用传入的端口号,没有传入,就使用默认端口号
	//int port = argc>=2 ? atoi(argv[1]):SER_PORT;
	
	// 填充服务器的地址信息结构体,给bind函数使用
	// 真实的地址信息结构体
	struct sockaddr_in sin;
	sin.sin_family  = AF_INET;  //必须填AF_INET
	sin.sin_port   = htons(SER_PORT);    //端口号的网络字节序
	sin.sin_addr.s_addr   = inet_addr(SER_IP);   // 本机IP的网络字节序
									// 127.0.0.1  本地环回IP,只能做本机通信
									// 0.0.0.0  代表运行环境中所有可用IP
	
	
	// 绑定服务器自身的地址信息
	
	if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin)) <0){
		fprintf(stderr,"__%d__",__LINE__);
		perror("bind");
		return -1;
	}
	printf("bind success __%d__\n",__LINE__);
	
	// 将套接字设置为被动监听状态
	if(listen(sfd,128)<0)            // 128 未完全连接的队列里面的客户机数量
	{
		fprintf(stderr,"__%d__",__LINE__);
		perror("listen");
		return -1;
	}
	printf("listen success __%d__\n",__LINE__);
	
	
	/*
	*  fd_set类型本质上是:
	typedef struct
	{
		long fds_bits[N]
	}fd_set;
	* 所以定义局部变量后,若不初始化,则是随机值
	* 有可能会随机到有效的但是不需要监测的文件描述符,导致程序bug
	* 所以 fd_set 定义的集合变量必须清空
	*
	*/
	
	// 创建读集合
	fd_set 	readfds,tempfds;
	FD_ZERO(&readfds);    // 清空集合
	FD_ZERO(&tempfds);
	
	//将需要的文件描述符添加到,合中
	FD_SET(0,&readfds);
	FD_SET(sfd,&readfds);
	
	int maxfd = sfd;     //集合中描述符最大的那个
	
	int s_res;
	int newfd =-1;	
	
	struct sockaddr_in savecin[1024]; //将客户端的地址信息存储到对应文件描述符下标位置

	
	while(1){
		tempfds = readfds;
		
		// 调用select函数,让内核检测集合中的文件描述符是否准备就绪
		s_res = select(maxfd+1, &tempfds,NULL,NULL,NULL);
		if(s_res < 0)
		{
			fprintf(stderr,"__%d__",__LINE__);
			perror("select");
			return -1;
		}
		else if(0 == s_res)
		{
			printf("time out...\n");
			break;
		}
		printf("__%d__\n",__LINE__);
		// 能运行到这个位置,则代表集合中有文件描述符准备就绪
		// 需要判断是集合中那个文件描述符准备就绪,走对应的处理函数即可
		// select 解除阻塞后,集合中会只剩下产生事件的文件描述符
		// 例如0号准备就绪,则集合中只会剩下0号文件描述符
		
		
		for(int i=0;i<=maxfd;i++)
		{
			//判断i代表的文件描述符是否在集合中
			if(FD_ISSET(i,&tempfds) == 0){
				continue;
			}
			//能运行到这个位置,则说明i代表的文件描述符在集合中
			if(0 == i)
			{
				printf("触发键盘输入事件__%d__\n",__LINE__);
				deal_keyboard_msg(readfds);
			}//判断sfd号描述符是否准备就绪
			else if(sfd == i)
			{
				printf("触发客户连接事件__%d__\n",__LINE__);
				deal_cliConnect(sfd,savecin,&readfds,&maxfd);
			}//判断newfd描述符是否准备就绪
			else
			{
				// 处理客户端交互事件
				printf("触发客户交互事件__%d__\n",__LINE__);
				deal_cliRecvSend(i,savecin,&readfds,&maxfd);
			
			}
		}
	}	
	// 关闭

	if(close(sfd)<0);    	// 监听newfd
	{
		perror("close");
		return -1;
	}
	return 0;
}	

// 处理客户端交互事件
int deal_cliRecvSend(int i,struct sockaddr_in savecin[],fd_set *preadfds,int *pmaxfd)
{
	char buf[128]="";
	ssize_t res =0;
	
	bzero(buf,sizeof(buf));  // memset(buf,0,sizeof(buf));
				//收   //会再这里等待。
		
	res = recv(i,buf,sizeof(buf),0);   // 返回收到的字符
	if(res<0){
		fprintf(stderr,"__%d__",__LINE__);
		perror("recv");
		return -1;
	}
	else if(0 == res){      // 说明客户端已经结束了
		printf("client offline __%d__\n",__LINE__);
		
		close(i);    //关闭文件描述符
		FD_CLR(i,preadfds);      // 将文件描述符中从集合删除
		
		//更行maxfd
		// 从目前记录的最大的maxfd开始依次往小的文件描述符判断是否在集合中
		while(FD_ISSET(*pmaxfd,preadfds) == 0 && (*pmaxfd)-->0);
		
		return 0;
	}
	printf("client newfd %d:%s\n",i,buf);
	// 打印收到的客户端的ip和端口
	printf("client port:%d  IP:%s\n",ntohs(savecin[i].sin_port),inet_ntoa(savecin[i].sin_addr));
		
	//发
	strcat(buf,"*_*");
	if(send(i, buf, sizeof(buf), 0)<0){
		fprintf(stderr,"__%d__",__LINE__);
		perror("send");
		return -1;
	}	
	return 0;
	
}
// 处理客户端连接事件
int deal_cliConnect(int sfd,struct sockaddr_in psavecin[],fd_set *peadfds,int *pmaxfd)
{
	struct sockaddr_in cin;  //  存储获取到的客户端地址信息	
	socklen_t addrlen = sizeof(cin);
	
	int newfd = accept(sfd,(struct sockaddr*)&cin,&addrlen);
	if(newfd<0)
	{
		fprintf(stderr,"__%d__",__LINE__);
		perror("accept");
		return -1;
	}
	printf("client accept success ,newfd = %d,__%d__\n",newfd,__LINE__);
			
	psavecin[newfd] = cin;
			
	// 将newfd添加到集合中,让select检测

	FD_SET(newfd,peadfds);   //让客户端连接成功后若要检测客户端的交互事件
							//需要将newfd添加到集合中,让内核监测
	*pmaxfd =*pmaxfd>newfd ? *pmaxfd:newfd;   // 更新maxfd

	return 0;
}

// 处理键盘输入事件,输入指定的文件描述符和内容
int deal_keyboard_msg(fd_set readfds)
{
	int sndfd;
	char buf[128];
	int res = scanf("%d %s",&sndfd,buf);
	while(getchar()!='\n');
	if(res != 2)
	{
		printf("请输入正确格式:fd string\n");
		return -1;
	}
	// 判断文件描述符是否合法
	if(sndfd <=3 || FD_ISSET(sndfd,&readfds) == 0)
	{
		printf("sndfd=%d 错误,请输入合法描述符\n",sndfd);
		return -1;
	}
	if(send(sndfd,buf,sizeof(buf),0)<0)
	{
		perror("send");
		return -1;
	}
	printf("send success\n");
	return 0;
	
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值