demo2的客户端和demo1的客户端相同只需要写重写服务端的程序.服务端代码如下:
#include<unistd.h>
#include<sys/stat.h>
#include<sys/wait.h>
#include<sys/types.h>
#include<fcntl.h>
#include<stdlib.h>
#include<stdio.h>
#include<errno.h>
#include<string.h>
#include<signal.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/socket.h>
int main(){
int sockfd=0;
sockfd=socket(AF_INET,SOCK_STREAM,0);
if(sockfd==-1){
perror("fun socket\n");
exit(0);
}
struct sockaddr_in serveraddr;
serveraddr.sin_family=AF_INET;
serveraddr.sin_port=htons(8001);
serveraddr.sin_addr.s_addr=inet_addr("127.0.0.1");
//serveraddr.sin_addr.s_addr=INADDR_ANY;//绑定本机的任意地址
int optval=1;
if(setsockopt(sockfd, SOL_SOCKET,SO_REUSEADDR,&optval, sizeof(optval))<0){
perror("setsockopt");
exit(0);
};
if(bind(sockfd, (const struct sockaddr *)&serveraddr,sizeof(serveraddr))<0){
perror("bind()");
exit(0);
};
//一旦调用了listen这个函数,那么这个套接字sockfd将变成被动套接字;只能接受连接,不能主动发送连接
//listen做了两个队列1、已经由客户端发出并达到服务器,服务器正在等待完成相应的tcp三次握手过程
//2、已完成连接的队列
if(listen(sockfd, SOMAXCONN)<0){
perror("listen()");
exit(0);
};
//
//struct sockaddr perraddr;//通用ip
//socklen_t perrlen;
struct sockaddr_in perraddr;//tcpip 地址结构
int perrlen=sizeof(perraddr);
int conn=0;
//这里是关键的代码,通过fork()出子进程,去处理其它的链接
while(1){
//accept 接受一个新的连接,这个新的连接是一个主动套接字
//while循环执行accept,如果listen维护完成的tcp三次握手的链接队列中不为空则则获取链接。
//一直到listen的链接中没有完成tcp三次握手的为止。阻塞在这里,不在循环
conn=accept(sockfd, (struct sockaddr *)&perraddr,&perrlen);
if(conn==-1){
perror("fun accept");
exit(0);
}
//打印客户端的信息
printf("perradd:%s\n perrport:%d\n",inet_ntoa(perraddr.sin_addr),ntohs(perraddr.sin_port));
//每来一个连接fork一个子进程
int pid=fork();
if(pid==0){
close(sockfd);//close 侦听套接字,因为子进程中不需要侦听。
char revbuf[100]={0};
//对字节维护的链接进行读和写。
while(1){
int ret=read(conn,revbuf,sizeof(revbuf));
if(ret==0){
//如果在读的过程中,对方已经关闭,已返回tcpip协议,会返回一个o数据报
printf("对方已关闭\n");
close(conn);//客户端关闭,副段也关系
}else if(ret<0){
printf("读数据失败");
exit(0);
}
fputs(revbuf,stdout);//服务器端收到数据,打印屏幕
write(conn,revbuf,sizeof(revbuf));
memset(revbuf,0,sizeof(revbuf));
}
}else if(pid>0){
close(conn);//父进程在侦听链接,不需要conn,关闭就可以了
}else{
close(conn);
close(sockfd);
printf("创建进程失败\n");
}
}
return 0;
}
服务端可以通过多进程处理链接了,如下:
多客户端通信