并发服务器模型,多线程并发

一、多线程并发完整代码

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

#define PORT 8808
#define IP "192.168.122.92"

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

void *callBack(void *arg);
void handle(int sig)
{
	while(waitpid(-1,NULL,WNOHANG) > 0);
}
int main(int argc, const char *argv[])
{
	if(signal(17,handle) == SIG_ERR)
	{
		perror("signal");
		return -1;
	}
	//创建流式套接字
	int sfd = socket(AF_INET,SOCK_STREAM,0);
	if(sfd < 0)
	{
		perror("socket");
		fprintf(stderr,"socket failed __%d__\n",__LINE__);
		return -1;
	}
	printf("socket success...\n");

	//设置允许端口号复用
	int reuse = 1;
	if(setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) < 0)  
	{
		perror("setsockopt");
		return -1;
	}

	//填充地址信息结构体,真是的地址信息结构体根据地址族制定
	struct sockaddr_in sin;
	sin.sin_family = AF_INET;
	sin.sin_port = htons(PORT);
	sin.sin_addr.s_addr = inet_addr(IP);

	//绑定服务器IP和端口号
	if(bind(sfd,(struct sockaddr *)&sin,sizeof(sin)) < 0)
	{
		perror("bind");
		fprintf(stderr,"bind failed __%d__\n",__LINE__);
		return -1;
	}
	printf("bind success...\n");

	//将套接字设置为被动监听状态
	if(listen(sfd,10) < 0)
	{
		perror("listen");
		return -1;
	}
	printf("listen success...\n");

	//从已完成连接的队列中获取一个客户信息,生成一个新的文件描述符,该文件描述符才是与客户端通信的文件描述符

	struct sockaddr_in cin;
	socklen_t addrlen = sizeof(cin);
	int newfd = -1;
	pthread_t tid;
	struct msg info;
	while(1)
	{
		newfd = accept(sfd,(struct sockaddr *)&cin,&addrlen);
		if(newfd < 0)
		{
			perror("accept");
			return -1;
		}
		printf("[%s : %d] accept success\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port));
		info.newfd = newfd;
		info.cin = cin;
		if(pthread_create(&tid,NULL,callBack,&info) !=  0)
		{
			fprintf(stderr,"pthread_create failed __%d__\n",__LINE__);
			return -1;
		}
		pthread_detach(tid);
	}
	//关闭所有文件描述符
	close(sfd);
	return 0;
}
void *callBack(void *arg)
{
	int newfd = ((struct msg *)arg)->newfd;
	struct sockaddr_in cin = ((struct msg *)arg)->cin;

	ssize_t res = 0;
	char buf[128] = "";
	while(1)
	{
		bzero(buf,sizeof(buf));
		res = recv(newfd,buf,sizeof(buf),0);
		if(res < 0)
		{
			perror("recv");
			return NULL;
		}
		else if(0 ==res)
		{
			printf("[%s : %d] client offline\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port));
			break;
		}
		printf("[%s : %d] %s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),buf);

		//发送数据
		if(send(newfd,buf,sizeof(buf),0) < 0)
		{
			perror("send");
			return NULL;
		}
	}
	close(newfd);
	pthread_exit(NULL);

}

二、多线程中的newfd,能否修改成全局

答:不行,newfd必须要另存,因为同一个进程下的线程共享其附属进程的所有资源,如果使用全局,则会导致每次连接客户端后, newfd和cin会被覆盖。

现象:

 由此可见,最后一次出错,客户端1被客户端2覆盖

三、多线程中分支线程的newfd能否不另存,直接用指针间接访问主线程的newfd

答:不行,如果使用指针间接访问外部成员变量,会导致,成员变量被覆盖,和上题类似。

猜你喜欢

转载自blog.csdn.net/weixin_53478812/article/details/132283748