socket 并发服务器 代码案例 (多进程模式)

    用Socket技术,完成一个简单的并发服务器,要求用fork()创建子进程来并发处理实现多进程并发,在接收到请求时,fork个进程,对父进程则关闭连接Socket,而子进程则关闭监听socket,每一个新的accept请求都由子进程来执行,而由父进程继续listen。 

server.c 端

#include<stdio.h>

#include<stdlib.h>
#include<strings.h>
#include<string.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>


#define PORT 1234
#define BACKLOG 2 //numbers of allowed connections
#define MAXDATASIZE 1000
void process_cli(int connectfd,struct sockaddr_in client);


void main()
{
    int listenfd,connectfd;
    pid_t pid;
    struct sockaddr_in server;
    struct sockaddr_in client;
    int sin_size;
    //创建socket关键字
    listenfd = socket(AF_INET,SOCK_STREAM,0);  
    if(listenfd == -1)
    {
        perror("creating socket failed");
        exit(1);
    }
    int opt = SO_REUSEADDR;
    setsockopt(listenfd,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt));
    bzero(&server,sizeof(server));
    server.sin_family = AF_INET;
    server.sin_port = htons(PORT);
    server.sin_addr.s_addr = htonl(INADDR_ANY);
    //绑定端口和IP地址
    if((bind(listenfd,(struct sockaddr *)&server,sizeof(struct sockaddr)))== -1)
    {
        perror("bind error\n");
        exit(1);
    }
    //监听客户端
    if(listen(listenfd,BACKLOG)==-1)
    {
        perror("listen error\n");
        exit(1);
    }
sin_size = sizeof(struct sockaddr_in);
while(1)
{
    //等待接收来自客户端的信息
    if((connectfd=accept(listenfd,(struct sockaddr*)&client,&sin_size))==-1)
    {
        perror("accept failed\n");
        exit(1);
    }
    //创建进程
    if((pid=fork())>0)
    {
        close(connectfd);
        continue;
    }
    else if(pid ==0)
    {
        close(listenfd);
        process_cli(connectfd,client);
      
        exit(1);
    }
    else
    {
        printf("fork error\n");
        exit(0);
    }
}
    close(listenfd);
}


void process_cli(int connectfd,struct sockaddr_in client)
{
    int num;
    char recvbuf[MAXDATASIZE],sendbuf[MAXDATASIZE],cli_name[MAXDATASIZE];
    printf("you got a connectfd from %s\n.",inet_ntoa(client.sin_addr));
    send(connectfd,"welcome to my server.\n",22,0);
    num=recv(connectfd,cli_name,MAXDATASIZE,0);
    if(num == 0)
    {
        close(connectfd);
        printf("client connects failed\n");
        return ;
    }
    cli_name[num-1]='\0';
    printf("client'name is %s\n",cli_name);


    while(num = recv(connectfd,recvbuf,MAXDATASIZE,0))
    {
        int i=0;
        recvbuf[num]='\0';
        printf("recevied client(%s)message:%s\n",cli_name,recvbuf);
        for(i=0;i<num;i++)
        {
            sendbuf[i] = recvbuf[num-i-2];
        }
        sendbuf[num-1]='\0';
        send(connectfd,sendbuf,strlen(sendbuf),0);
    }
    close(connectfd);
}

client.c 端

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<netdb.h>


#define PORT 1234
#define MAXDATASIZE 1000


int main(int argc,char *argv[])
{
    int fd,numbytes;
    char buf[MAXDATASIZE];
    struct hostent *he;
    struct sockaddr_in server;
    int i=1;
    if((he=gethostbyname(argv[1]))==NULL)
    {
        perror("gethostbyname error");
        exit(1);
    }
    if((fd=socket(AF_INET,SOCK_STREAM,0))==-1)
    {
        perror("socket failed");
        exit(1);
    }
    bzero(&server,sizeof(server));
    server.sin_family =AF_INET;
    server.sin_port =htons(PORT);
    server.sin_addr =*((struct in_addr*)he->h_addr);
    if(connect(fd,(struct sockaddr*)&server,sizeof(struct sockaddr))==-1)
    {
        perror("bind error");
        exit(1);
    }
    if((numbytes=recv(fd,buf,MAXDATASIZE,0))==-1)
    {
        perror("recv error");
        exit(1);
    }
    buf[numbytes]='\0';
    printf("%s\n",buf);
    printf("input your name:");
    scanf("%s",buf);
    if((numbytes=send(fd,buf,strlen(buf),0))==-1)
    {
        perror("send failed");
        exit(1);
    }
    while(i)
    {
        printf("input me message(max char:%d):",MAXDATASIZE);
        scanf("%s",buf);
        if(strlen(buf)<1)
        {
            i=0;
        }
        if((numbytes=send(fd,buf,strlen(buf),0))==-1)
        {
            perror("send failed");
            exit(1);
        }


        if((numbytes=recv(fd,buf,MAXDATASIZE,0))==-1)
        {   
            perror("recv error");
            exit(1);
        }
        buf[numbytes]='\0';
        printf("server message:%s\n",buf);
        printf("\n");
    }
    close(fd);
    return 0;
}

运行结果可以多客户端对应一个服务器端。





  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值