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);
}