网络编程6——多线程并发服务器实现(线程分离

recall一下
在这里插入图片描述

代码实现

(还是要先封装wrap.h)
1,socket
2,bind,绑定端口,注意参数,定义初始化强转化
3,listen函数,限定同时访问数
4,loop中,accept等待连接,注意参数,第三个参数长度的类型是socklen_t

#include<stdio.h>
#include<string.h>
#include<arpa/inet.h>
#include<pthread.h>
#include<ctype.h>
#include<unistd.h>
#include<fcntl.h>
#include "wrap.h"

#define MAXLINE 8192
#define SERV_PORT 8000

struct s_info//定义一个结构体,将地址结构跟cfd捆绑
{
	struct sockaddr_in cliaddr;
	int cfd;
}

void *do_work(void *arg)//回调函数  传入的参数是结构体  
{//回调函数中做大小写转换
	int n, i;
	struct s_info *ts = (struct s_info*)arg;//接收参数,要注意参数类型
	char buf[MAXLINE];//读写需要的缓冲区
	char str[INET_ADDRSTRLEN];
	while(1)
	{
		n = Read(ts->cfd,buf,MAXLINE);//从客户端读入
		if(n == 0)//读完了
		{
			printf("the client %d closed... "ts->cfd);
			break;
		}
		printf("receive from %s at port %d...\n",inet_ntop(AF_INET, &(*ts).cliaddr.sin_addr,str,sizeof(str)),ntohs((*ts).cliaddr.sin_port));//可以读出客户端的套接字
		//转换大小写
		for(i = 0; i<n; i++)
		{
			buf[i] = toupper(buf[i]);
			Write(STDOUT_FILENO,buf,n);//写到屏幕
			Write(ts->cfd, buf, n);
		}
	}
		Close(ts->cfd);
		return (void*)0;	
}

int main(void)
{
	int lfd,cfd;
	struct sockaddr_in servaddr,cliaddr;
	socklen_t cliaddr_len;
	struct s_info ts[256];//创建结构体数组
	int i;
	bzero(&servaddr, sizeof(servaddr));//地址结构清零
	servaddr.sin_family = AF_INET;
	servaddr.sin_addr.s_addr = htonl(INADDR_ANY);//指定本地任意IP
	servaddr.sin_port = htons(SERV_PORT);//指定端口号
	lfd = Socket(AF_INET, SOCK_STREAM,0);//①创建一个socket,得到lfd
	Bind(lfd, (struct sockaddr *)&servaddr, sizeof(servaddr));//②
	Listen(lfd,128);//③
	//④accept等连接
	while(1)
	{
		cliaddr_len = sizeof(cliaddr);//这个类型要注意!
		cfd = Accept(lfd, (struct sockaddr*)&cliaddr, &cliaddr_len);//阻塞监听客户端链接请求
		//结构体数组的意义/意思?
		ts[i].cliaddr = cliaddr;
		ts[i].cfd = cfd;
		//接收之后开始创建子线程?
		pthread_create(&tid,NULL,do_work, (void*)&ts[i]);//tid是创建的线程的地址,do_work是回调函数,将ts结构体作为参数传给回调函数
		pthread_detach(tid);//子线程分离,防止僵尸线程产生⭐
		i++;	
	}
	return 0;
}
原创文章 119 获赞 27 访问量 5981

猜你喜欢

转载自blog.csdn.net/qq_37299596/article/details/105967839