Socket编程---c/s模型demo2

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

服务端可以通过多进程处理链接了,如下:
在这里插入图片描述
多客户端通信

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值