2月15日 并发tcp服务器

文章提供了两个示例,分别展示了使用多进程和多线程技术实现的TCP并发服务器。在多进程方案中,服务器接收到新的客户端连接后,通过fork创建子进程来处理。而在多线程方案中,服务器创建线程来处理客户端交互。两种方法都用于处理客户端的并发请求,但实现机制不同。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.多进程并发服务器代码

#include<stdio.h>

#include<sys/types.h>

#include<sys/socket.h>

#include<arpa/inet.h>

#include<netinet/in.h>

#include<string.h>

#include<unistd.h>

#include<sys/wait.h>

#include<signal.h>

#include<stdlib.h>

#define ERR_MSG(msg) do{\

fprintf(stderr,"line:_%d_",__LINE__);\

perror(msg);\

}while(0)

typedef void (*sighandler_t)(int);

int deal_cli_inter(int newfd,structsockaddr_in cin);

void handler(int sig)

{

while(waitpid(-1,NULL,WNOHANG)>0);

}

int main(int argc, const char *argv[])

{

sighandler_ts=signal(SIGCHLD,handler);

if(SIG_ERR==s){

perror("signal");

return-1;

}

//创建流式套接字-TCP

intsfd = socket(AF_UNIX,SOCK_STREAM,0);

if(sfd<0){

ERR_MSG("socket");

return-1;

}

printf("sfd=%d\n",sfd);

if(access("./unix",F_OK)==0){

if(unlink("./unix")<0){

perror("unlink");

return-1;

}

}

//填充地址信息结构体,真实地址信息结构体在man 7 IP

//AF-INET ---》man 7 IP

structsockaddr_un sun;

sin.sun_family= AF_UNIX; //必须填AF_INET;

sin.sin_port =htons(PORT); //网络字节序

//功能:将IP地址和端口号绑定到制定套接字中;

if(bind(sfd,(structsockaddr*)&sun,sizeof(sun))<0)

{

ERR_MSG("bind");

return-1;

}

printf("bindsuccess _%d_\n",__LINE__);

//功能:将套接字设置为被动监听状态,只负责监听是否有客户端连接成功;

if(listen(sfd,10)<0){

ERR_MSG("listen");

return-1;

}

printf("listensuccess\n");

structsockaddr_in cin; //存储客户端的地址信息

socklen_taddrlen=sizeof(cin);

intnewfd=0;

//功能:阻塞函数,等待客户端链接成功。

//当客户端链接成功后,会从已完成连接的队列头获取一个客户信息

//并生成一个新的文件描述符;

//注意:新生成的文件描述符才是用于通信的文件描述符

while(1)

{

newfd=accept(sfd,(structsockaddr*)&cin,&addrlen);

if(newfd<0){

ERR_MSG("accept");

return-1;

}

printf("%s|%dnewfd=%d\n",newfd);

pid_tcpid;

cpid=fork();

if(cpid==0){

close(sfd);

deal_cli_inter(newfd,cin);

exit(4);

}

if(cpid>0){

close(newfd);

}

else{

perror("fork");

return-1;

}

}

close(sfd);

return0;

}

int deal_cli_inter(int newfd,structsockaddr_in cin)

{

charbuf[128]="";

ssize_tres=0;

while(1)

{

bzero(buf,sizeof(buf));

res=recv(newfd,buf,sizeof(buf),0);

if(res<0){

ERR_MSG("recv");

return-1;

}

elseif(0==res)

{

printf("[%s|%d]newfd=%d",newfd);

break;

}

printf("[%s|%d]newfd=%d:%s\n",newfd,buf);

//发送-->谁发给我数据,我就发还给谁

strcat(buf,"*-*");

if(send(newfd,buf,sizeof(buf),0)<0)

{

ERR_MSG("send");

return-1;

}

printf("sendsuccess\n");

}

close(newfd);

return0;

}

2.多线程并发服务器代码

#include<stdio.h>

#include<sys/types.h>

#include<sys/socket.h>

#include<arpa/inet.h>

#include<netinet/in.h>

#include<string.h>

#include<unistd.h>

#include<sys/wait.h>

#include<signal.h>

#include<stdlib.h>

#include<pthread.h>

#define ERR_MSG(msg) do{\

fprintf(stderr,"line:_%d_",__LINE__);\

perror(msg);\

}while(0)

#define PORT 9999 //1024~49151

#define IP "192.168.8.99" //ifconfig查看

//定义结构体,用于向分线程传参

struct info{

intnewfd;

structsockaddr_in cin;

};

//声明 deal_cli_inter函数

void* deal_cli_inter(void* arg);

int main(int argc, const char *argv[])

{

//创建流式套接字-TCP

intsfd = socket(AF_INET,SOCK_STREAM,0);

if(sfd<0){

ERR_MSG("socket");

return-1;

}

printf("sfd=%d\n",sfd);

//填充地址信息结构体,真实地址信息结构体在man 7 IP

//AF-INET ---》man 7 IP

structsockaddr_in sin;

sin.sin_family= AF_INET; //必须填AF_INET;

sin.sin_port =htons(PORT); //网络字节序

sin.sin_addr.s_addr=inet_addr(IP);//本机IP,ifconfig查看

//允许端口快速被复用

intreuse=1;

if(setsockopt(sfd,SOL_SOCKET,SO_REUSEADDR,&reuse,sizeof(reuse))<0)

{

perror("setsockopt");

return-1;

}

//功能:将IP地址和端口号绑定到制定套接字中;

if(bind(sfd,(structsockaddr*)&sin,sizeof(sin))<0)

{

ERR_MSG("bind");

return-1;

}

printf("bindsuccess _%d_\n",__LINE__);

//功能:将套接字设置为被动监听状态,只负责监听是否有客户端连接成功;

if(listen(sfd,10)<0){

ERR_MSG("listen");

return-1;

}

printf("listensuccess\n");

structsockaddr_in cin; //存储客户端的地址信息

socklen_taddrlen=sizeof(cin);

intnewfd=0;

pthread_ttid;

structinfo cliMsg; //创建传参的结构体变量

//功能:阻塞函数,等待客户端链接成功。

//当客户端链接成功后,会从已完成连接的队列头获取一个客户信息

//并生成一个新的文件描述符;

//注意:新生成的文件描述符才是用于通信的文件描述符

while(1)

{

newfd=accept(sfd,(structsockaddr*)&cin,&addrlen);

if(newfd<0){

ERR_MSG("accept");

return-1;

}

printf("%s|%dnewfd=%d\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),newfd);

cliMsg.newfd=newfd;

cliMsg.cin=cin;

//能运行到当前位置,代表客户端连接成功,且accept 处理完毕

//此时需要创建一个分支线程,用于与客户端交互

if(pthread_create(&tid,NULL,deal_cli_inter,&cliMsg)!=0)

{

fprintf(stderr,"pthread_createfailed\n");

return-1;

}

pthread_detach(tid);

}

close(sfd);

return0;

}

void* deal_cli_inter(void* arg)

{

intnewfd=((struct info*)arg)->newfd;

structsockaddr_in cin=((struct info*)arg)->cin;

charbuf[128]="";

ssize_tres=0;

while(1)

{

bzero(buf,sizeof(buf));

res=recv(newfd,buf,sizeof(buf),0);

if(res<0){

ERR_MSG("recv");

returnNULL;

}

elseif(0==res)

{

printf("[%s|%d]newfd=%d客户断开连接\n",

inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),newfd);

break;

}

printf("[%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)

{

ERR_MSG("send");

returnNULL;

}

printf("sendsuccess\n");

}

close(newfd);

pthread_exit(NULL);

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值