socket之tcp多进程客户服务器编程

实现基于TCP的C/S服务模型,服务器将客户端的输入回显给客户端。

服务器:

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

static int startup(const char *_ip,int _port)
{

	int sock = socket(AF_INET,SOCK_STREAM,0);
	if(sock < 0)
	{
		perror("socket");
		exit(2);
	}
	
	int on = 1;
	int ret = setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on));
	if(ret < 0)
	{
		perror("setsockopt");
		exit(5);
	}

	struct sockaddr_in serv_addr;
	serv_addr.sin_family = AF_INET;
	serv_addr.sin_port = htons(_port);
	serv_addr.sin_addr.s_addr = inet_addr(_ip);

	if(bind (sock,(struct sockaddr*)& serv_addr,sizeof(serv_addr) )  < 0     )
	{
		perror("bind");
		exit(3);
	}
	if(listen(sock,10) < 0)
	{
		perror("listen");
		exit(4);
	}

	return sock;
}

static void usage(const char *str)
{
	printf("Usage:%s [IP] [PORT]\n",str);
}

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

	if(argc != 3)
	{
		usage(argv[0]);
		return 1;
	}

	int listen_sock = startup(argv[1],atoi(argv[2]));

	struct sockaddr_in clie_addr;
	socklen_t len = sizeof(clie_addr);

	while(1)
	{
		int sock = accept(listen_sock ,(struct sockaddr*)&clie_addr,&len );
		if(sock > 0)
		{
			pid_t id = fork();

			if(id == 0) //child
			{
				if(fork() > 0)//子进程退出,让孙子进程办事,孙子进程成为孤儿进程,被1号进程回收。
					exit(0); 

				char buf[BUFSIZ];

				while(1)
				{
					ssize_t s = read(sock,buf,sizeof(buf)-1);
					if(s > 0)
					{
						buf[s] = 0;
						printf("client say#:%s\n",buf);
						write(sock,buf,strlen(buf));
					}
					else if(s == 0)
					{
						printf("client quit!\n");
						break;
					}
					else
						break;
				}
			}
			else//father 
			{
				close(sock); //父进程关闭已连接套接字,留下监听套接字。
				//if(fork() > 0)
				//	exit(0); //父进程退出,留下两个子进程(孤儿进程,1号进程回收资源,不会浪费资源),一个子进程负责监听,一个负责数据传输。
			}

		}
		else
		{
			perror("accept");
		}
	}


	return 0;
}


客户端:

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


static void usage(const char *str)
{
	printf("Usage:%s,[IP],[PORT]\n",str);
}

int main(int argc ,char *argv[])
{
	int sock = socket(AF_INET,SOCK_STREAM,0);
	if(sock < 0)
	{
		perror("sock");
		return 1;
	}

	struct sockaddr_in serv_addr;
	serv_addr.sin_family = AF_INET;
	serv_addr.sin_port = htons(atoi(argv[2]));
	serv_addr.sin_addr.s_addr = inet_addr(argv[1]);

	if(connect(sock,(struct sockaddr*)& serv_addr,sizeof(serv_addr)) < 0)
	{
		perror("connect");
		return 3;
	}

	char buf[BUFSIZ];
	while(1)
	{

		printf("client say# ");
		fflush(stdout);

		ssize_t s = read(0,buf,sizeof(buf) - 1);
		if(s > 0)
		{
			buf[s - 1] = 0;
			write(sock,buf,strlen(buf));
			ssize_t s2 = read(sock,buf,sizeof(buf) -1 );
			if(s2 > 0)
			{
				buf[s2] = 0;
				printf("server echo # %s\n",buf);

			}
			
		}

	}

	return 0;
}

截图:

客户端1:


客户端2:


服务器:



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值