网络编程第四天

1.多进程实现并发服务器

运行结果:

网络编程第四天_第1张图片

代码:

#include 

#define PORT 8888 				//1024~49151
#define IP "192.168.124.37" 	//ifconfig本机IP

int func(int newfd, struct sockaddr_in cin);

//回收子进程资源
void handler(int sig)
{
	while(waitpid(-1, NULL, WNOHANG) > 0);
}

/*
 * function:    多进程实现并发服务器
 * @param [ in] 
 * @param [out] 
 * @return      
 */
int main(int argc, const char *argv[])
{
	//接收17号信号
	if(signal(17, handler) == SIG_ERR)
	{
		ERR_MSG("signal");
		return -1;
	}
	
	//创建流式套接字
	int sfd = socket(AF_INET,SOCK_STREAM,0);
	if(sfd < 0)
	{
		ERR_MSG("socket");
		return -1;
	}
	printf("socket create success sfd = %d __%d__\n",sfd,__LINE__);

	//允许端口快速被复用---监测端口号是否真的有进程在占用,如果没有,则快速复用
	int reuse = 1;
	if(setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) < 0)
	{
		ERR_MSG("setsockopt");
		return -1;
	}
	printf("允许端口快速复用成功\n");


	//定义并填充地址消息结构体,给服务绑定使用
	//真实的地址信息结构体根据地址族指定,AT_INET: man 7 ip
	struct sockaddr_in sin;
	sin.sin_family 		= AF_INET; 		//必须为AF_INET
	sin.sin_port 		= htons(PORT); 	//端口号的网络字节序, 1024~49151
	sin.sin_addr.s_addr = inet_addr(IP);//本机IP的网络字节序

	//绑定服务器自身的地址消息 -->必须绑定
	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("lesten");
		return -1;
	}
	printf("listen success __%d__\n",__LINE__);

	struct sockaddr_in cin; 	//接收客户端的地址消息
	socklen_t addrlen = sizeof(cin);

	//获取连接成功的客户端信息,生成一个新的套接字
	while(1)
	{
		int newfd = accept(sfd,(struct sockaddr*)&cin,&addrlen);
		if(newfd < 0)
		{
			ERR_MSG("accept");
			return -1;
		}
		//打印客户端的IP和端口
		printf("[%s : %d]newfd = %d 客户端连接成功__%d__\n",\
				inet_ntoa(cin.sin_addr) , ntohs(cin.sin_port),newfd,__LINE__);

		pid_t cpid = fork();
		//子进程只做收发
		if(0 == cpid)
		{
			close(sfd);
			func(newfd, cin);
			exit(0);
		}
		//父进程只做连接
		else if(0 < cpid)
		{
			close(newfd);
		}
		else
		{
			ERR_MSG("fork");
			return -1;
		}

	}
	//关闭所有文件描述符
	if(close(sfd) < 0)
	{
		ERR_MSG("close");
		return -1;
	}
	return 0;
}

int func(int newfd, struct sockaddr_in cin)
{
	//接收缓冲
	char buf_r[128] = "";
	//发送缓冲
	char buf_s[128] = "";
	ssize_t res = 0;
	while(1)
	{
		bzero(buf_r,sizeof(buf_r));
		//接收客户端数据
		res = recv(newfd, buf_r, sizeof(buf_r), 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_r,__LINE__);

		//向客户端发送数据 ---> //从终端获取数据
		bzero(buf_s,sizeof(buf_s));
		//strcat(buf_s,"^_^");
		printf("请输入>>>");
		fgets(buf_s,sizeof(buf_s),stdin);
		buf_s[strlen(buf_s)-1] = '\0';

		if(send(newfd, buf_s, sizeof(buf_s), 0) < 0)
		{
			ERR_MSG("send");
			return -1;
		}
		printf("send success __%d__\n", __LINE__);
	}
	if(close(newfd) < 0)
	{
		ERR_MSG("close");
		return -1;
	}

}

2.多线程实现并发服务器

运行结果:

网络编程第四天_第2张图片

代码:

#include 

#define PORT 8888 				//1024~49151
#define IP "192.168.170.126" 	//ifconfig本机IP

void* deal_cli_msg(void* arg);
	
struct climsg
{
	int newfd;
	struct sockaddr_in cin;
};

/*
 * function:    多线程实现并发服务器
 * @param [ in] 
 * @param [out] 
 * @return      
 */
int main(int argc, const char *argv[])
{
	//创建流式套接字
	int sfd = socket(AF_INET,SOCK_STREAM,0);
	if(sfd < 0)
	{
		ERR_MSG("socket");
		return -1;
	}
	printf("socket create success sfd = %d __%d__\n",sfd,__LINE__);

	//允许端口快速被复用---监测端口号是否真的有进程在占用,如果没有,则快速复用
	int reuse = 1;
	if(setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) < 0)
	{
		ERR_MSG("setsockopt");
		return -1;
	}
	printf("允许端口快速复用成功\n");


	//定义并填充地址消息结构体,给服务绑定使用
	//真实的地址信息结构体根据地址族指定,AT_INET: man 7 ip
	struct sockaddr_in sin;
	sin.sin_family 		= AF_INET; 		//必须为AF_INET
	sin.sin_port 		= htons(PORT); 	//端口号的网络字节序, 1024~49151
	sin.sin_addr.s_addr = inet_addr(IP);//本机IP的网络字节序

	//绑定服务器自身的地址消息 -->必须绑定
	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("lesten");
		return -1;
	}
	printf("listen success __%d__\n",__LINE__);

	//struct sockaddr_in cin; 	//接收客户端的地址消息
	struct climsg info;
	socklen_t addrlen = sizeof(info.cin);
	pthread_t tid;

	while(1)
	{
		//获取连接成功的客户端信息,生成一个新的套接字
		info.newfd = accept(sfd,(struct sockaddr*)&info.cin,&addrlen);
		if(info.newfd < 0)
		{
			ERR_MSG("accept");
			return -1;
		}
		//打印客户端的IP和端口
		printf("[%s : %d]newfd = %d 客户端连接成功__%d__\n",\
				inet_ntoa(info.cin.sin_addr) , ntohs(info.cin.sin_port),info.newfd,__LINE__);

		//创建线程
		if(pthread_create(&tid, NULL, deal_cli_msg, (void*)&info) != 0)
		{
			fprintf(stderr,"pthread_create error\n");
			return -1;
		}
		//分离线程线程退出后系统自动回收资源
		pthread_detach(tid);
	}
	//关闭所有文件描述符
	if(close(sfd) < 0)
	{
		ERR_MSG("close");
		return -1;
	}
	return 0;
}

//多线程收发数据
void* deal_cli_msg(void* arg)
{
	int newfd = ((struct climsg*)arg)->newfd;
	struct sockaddr_in cin = ((struct climsg*)arg)->cin;
	//接收缓冲
	char buf_r[128] = "";
	//发送缓冲
	char buf_s[128] = "";
	ssize_t res = 0;
	while(1)
	{
		bzero(buf_r,sizeof(buf_r));
		//接收客户端数据
		res = recv(newfd, buf_r, sizeof(buf_r), 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_r,__LINE__);
		
		//向客户端发送数据 ---> //从终端获取数据
		bzero(buf_s,sizeof(buf_s));
		//strcat(buf_s,"^_^");
		printf("请输入>>>");
		fgets(buf_s,sizeof(buf_s),stdin);
		buf_s[strlen(buf_s)-1] = '\0';

		if(send(newfd, buf_s, sizeof(buf_s), 0) < 0)
		{
			ERR_MSG("send");
			break;
		}
		printf("send success __%d__\n", __LINE__);
	}
	
	//关闭套接字
	close(newfd);
	//退出线程
	pthread_exit(NULL);
}

你可能感兴趣的:(网络编程,网络,c语言,linux,服务器)