多线程

多线程

c 需要通过线程创建函数以值传递的方式传递函数线程。close( c ) 只能函数线程结束服务后调用,主线程不敢调用

  1. 主线程完成与客户端的连接工作,函数线程负责处理客户链接
  2. 线程如何拿到与客户连接的文件描述符?
    创建线程时,以值传递的方式传递给函数线程。(不能使用地址传递)
  3. 主线程不需要关闭文件描述符,同一个进程中所有线程共享进程资源仅有独立的栈区
  4. 不需要处理僵死进程,本身不会有子进程

缺陷

1、系统能够进程或者一个进程中能够创建的线程都是有限,
2、为一个客户端连接创建一个进程或者线程,客户端断开则销毁是不划算的

多进程多线程比较

1、代码控制方面:多线程更轻便,多进程更复杂
2、编程角度:多线程比多进程实现简单,控制简单
3、占据资源方面:多进程比多线程大
4、切换:线程切换比进程快
5、资源共享:线程比进程共享的资源多。线程共享全局变量、堆区变量、文件描述符,进程共享文件描述符。
6、线程因为共享的资源太多,他的安全性要低。
7、数量:多进程比多线程要多很多。一个进程能创建线程的数量是有限的,而且这个数量远远比一个系统上能执行的进程数量要小得多

Sever

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

void*CommClient(void *data)
{
	int c=(int)data;
	while(1)
	{
		char buff[128]={0};
		int n=recv(c,buff,127,0);//阻塞  有数据可读或者客户端退出都会返回
		if(n==0)
		{
			printf("client unlink\n");
			close(c);
			break;//要继续接收下一个连接
		}
		else if(n==-1)
		{
			printf("error\n");
			close(c);
			break;
		}

		printf("n==%d: %s",n,buff);

		send(c,"OK",2,0);
	}
}

int main()
{
	int listenfd=socket(AF_INET,SOCK_STREAM,0);
	assert(-1!=listenfd);

	struct sockaddr_in ser,cli;
	memset(&ser,0,sizeof(ser));//清空
	ser.sin_family=AF_INET;//IPv4地址族
	ser.sin_port=htons(6000);//主机到网络的short型,给服务器指定端口号
	inet_aton("127,0,0,1",(struct in_addr*)&ser.sin_addr);//把字符型地址转换为整形

	int res=bind(listenfd,(struct sockaddr*)&ser,sizeof(ser));
	assert(-1!=res);

	listen(listenfd,5);
	while(1)
	{
		int len=sizeof(cli);
		//c特定客户端和服务器连接的文件描述符
		//accept函数只有在有客户端连接的情况下返回
		//c 需要通过线程创建函数以值传递的方式传递函数线程。
		int c=accept(listenfd,(struct sockaddr*)&cli,&len);//阻塞运行
		if(c<0)
		{
			printf("link error");
			continue;
		}
		pthread_t id;
		int res=pthread_create(&id,NULL,CommClient,(void *)c);
		assert(res==0);
	}
	close(listenfd);
}

Cli

#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <unistd.h>
#include <string.h>
#include <netinet/in.h>//字节序列转换函数所用
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>//地址转换函数所用

int main()
{
	int sockfd=socket(AF_INET,SOCK_STREAM,0);
	assert(-1!=sockfd);

	struct sockaddr_in ser;
	memset(&ser,0,sizeof(ser));
	ser.sin_family=AF_INET;
	ser.sin_port=htons(6000);
	inet_aton("127.0.0.1",(struct in_addr*)&ser.sin_addr);

	int res=connect(sockfd,(struct sockaddr*)&ser,sizeof(ser));
	assert(-1!=res);
	
	while(1)
	{
		printf("please input: ");
		char data[128]={0};
		fgets(data,128,stdin);
		if(strncmp(data,"bye",3)==0)
		{
			close(sockfd);
			break;
		}

		send(sockfd,data,strlen(data)+1,0);
		char buff[128]={0};
		int n=recv(sockfd,buff,127,0);
		if(n<=0)
		{
			printf("error\n");
			close(sockfd);
			break;
		}
		printf("n==%d: %s\n",n,buff);
	}
}

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值