【问题标题】:Multiclient server using fork()使用 fork() 的多客户端服务器
【发布时间】:2012-11-20 02:32:05
【问题描述】:

我正在尝试使用 fork() 创建一个套接字编程服务器来同时处理多个客户端。 但我无法正确实施。我已经尝试了很长时间。 我面临的问题是 1)解决绑定问题 2)问题如何处理父进程和子进程 3)如何结束服务器程序即..返回控制台 我的单个客户端服务器的程序运行正常。这是我的多个客户端服务器的代码。

#include<signal.h>
#include<stdio.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<string.h>
#include<sys/types.h>
#include<stdlib.h>

int main()
{

    struct sockaddr_in myaddr ,clientaddr;
    int sockid,newsockid;
    sockid=socket(AF_INET,SOCK_STREAM,0);
    memset(&myaddr,'0',sizeof(myaddr));
    myaddr.sin_family=AF_INET;
    myaddr.sin_port=htons(8888);
    myaddr.sin_addr.s_addr=inet_addr("127.0.0.1");
    if(sockid==-1)
    {
        perror("socket");
    }
    int len=sizeof(myaddr);
    if(bind(sockid,( struct sockaddr*)&myaddr,len)==-1)
    {
        perror("bind");
    }
    if(listen(sockid,10)==-1)
    {
        perror("listen");
    }
    int pid,new;
    static int counter=0;
    for(;;)
    {       a:
            new =accept(sockid,(struct sockaddr *)&clientaddr,&len);

            if(pid=fork()==-1)
            {
                close(new);
                continue;

            }   
            else if(pid>0)
            {
            counter++;
            //wait();
            goto a;
            printf("here2");
            //close(new);
            continue;
            }   
            else if(pid==0)
            {
            counter++;
            printf("here 1");
            send(new,"hi",100,0);
            send(new,(char *) &counter,1,0);

            //kill(pid,SIGKILL);
            //close(new);
            }

    }
    printf("here3");
    close(sockid);
    return 0;
}

这是简单的客户端程序

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


int main()
{

    struct sockaddr_in myaddr ,serveraddr;
    int sockid;
    sockid=socket(AF_INET,SOCK_STREAM,0);
    memset(&myaddr,'0',sizeof(myaddr));
    myaddr.sin_family=AF_INET;
    myaddr.sin_port=htons(8888);
    myaddr.sin_addr.s_addr=inet_addr("127.0.0.1");
    if(sockid==-1)
    {
        perror("socket");
    }
    int len=sizeof(myaddr);
    if(connect(sockid,(const struct sockaddr*)&myaddr,len)==-1)
    {
        perror("connect");
    }
    fprintf(stdout,"Client Online....");
    char s[10000];


        //gets(s);
        //send(sockid,s,10000,0);
        recv(sockid,&s,10000,0);
        fprintf(stdout,"Server says....");
        puts(s);
        recv(sockid,&s,10000,0);
        fprintf(stdout,"Server says....");
        puts(s);

sleep(10);  
    close(sockid);
    return 0;
}

有人可以告诉我我做错了什么以及正确的方法是什么..?任何帮助将不胜感激...

【问题讨论】:

  • 帮帮我,我已经厌倦了尝试
  • @VictorSorokin:如果你能看看我的具体程序或我的目的,那就太好了。我已经写过我对单个客户端服务器没有问题。
  • "memset(&myaddr,'0',sizeof(myaddr));"是一个有趣的错字:)

标签: c algorithm sockets unix fork


【解决方案1】:

您遇到的主要问题是== 的优先级高于=,所以这一行:

if(pid=fork()==-1)

fork() == -1 的结果分配给pid,这不是你想要的:当fork() 成功时,它总是0,在孩子父母。你需要使用:

if((pid = fork()) == -1)

您还应该在fork() 之后的父级中close(new) - 子级现在拥有该套接字。如果要发送计数器的文本版本,需要使用snprintf()将其转换为文本。孩子也应该在完成后退出 - 在您的代码中执行此操作的最简单方法是跳出循环。经过这些更正后,您的服务器中的内部循环如下所示:

for(;;)
{
        new = accept(sockid, (struct sockaddr *)&clientaddr, &len);

        if ((pid = fork()) == -1)
        {
            close(new);
            continue;
        }
        else if(pid > 0)
        {
            close(new);
            counter++;
            printf("here2\n");
            continue;
        }
        else if(pid == 0)
        {
            char buf[100];

            counter++;
            printf("here 1\n");
            snprintf(buf, sizeof buf, "hi %d", counter);
            send(new, buf, strlen(buf), 0);
            close(new);
            break;
        }
}

【讨论】:

  • 子进程也可以和/或应该关闭 sockid 吗?我问,因为我遇到了一个问题,我有一个子进程在循环中运行,然后杀死父进程并尝试重新启动它,但是它无法绑定套接字,因为子进程仍然绑定了服务器套接字。
  • @NHDaly:是的,如果孩子要长寿的话。我没有把它放在这里的代码中,因为我们知道孩子无论如何都会退出。
  • 好的,谢谢。这就是我一直在尝试做的事情,但是由于某种原因,一旦子进程退出,随后对 accept(sockid,...) 的调用就会失败。我不太清楚为什么.. perror 告诉我:failed to accept: Interrupted system call.
  • NHDaly:那只是因为在子退出时发送给父级的SIGCHLD 中断了accept()EINTR 应被视为暂时失败 - 在这种情况下,只需循环并重试 accept()
  • 啊,谢谢。问题是我实际上在每个循环开始时接收到两个连接,所以当第一个连接失败时我总是被一个连接关闭。谢谢。
【解决方案2】:

试试这个也许你会得到解决方案,

服务器程序:

#include <sys/types.h>
#include <sys/socket.h>
#include <stdio.h>
#include <netinet/in.h>
#include <signal.h>
#include <unistd.h>

int main()
{
int server_sockfd, client_sockfd;
int server_len, client_len;
struct sockaddr_in server_address;
struct sockaddr_in client_address;

server_sockfd = socket(AF_INET, SOCK_STREAM, 0);

server_address.sin_family = AF_INET;
server_address.sin_addr.s_addr = htonl(INADDR_ANY);
server_address.sin_port = htons(9734);
server_len = sizeof(server_address);
bind(server_sockfd, (struct sockaddr *)&server_address,server_len);

/* Create a connection queue, ignore child exit details and wait for
clients. */

listen(server_sockfd, 5);

signal(SIGCHLD, SIG_IGN);

while(1) {
char ch;

printf("server waiting\n");

/* Accept connection. */

client_len = sizeof(client_address);
client_sockfd = accept(server_sockfd,(struct sockaddr *)&client_address, &client_len);

/* Fork to create a process for this client and perform a test to see
whether we're the parent or the child. */

if(fork() == 0) {

/* If we're the child, we can now read/write to the client on
client_sockfd.
The five second delay is just for this demonstration. */

read(client_sockfd, &ch, 1);
sleep(5);
ch++;
write(client_sockfd, &ch, 1);
close(client_sockfd);
exit(0);
}

/* Otherwise, we must be the parent and our work for this client is
finished. */

else {
close(client_sockfd);
}
}
}

客户程序:

#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<string.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<netinet/in.h>
void main()
{
    int sid;
    char s[10]={},s1[10]={};
    struct sockaddr_in ssock,csock;
    sid=socket(AF_INET,SOCK_STREAM,0);
    ssock.sin_family=AF_INET;
    ssock.sin_addr.s_addr=inet_addr("127.0.0.1");
    ssock.sin_port=htons(9734);
    connect(sid,(struct sockaddr *)&ssock,sizeof(ssock));
    while(1)
    {
        printf("\n Enter the string:");
        scanf("%s",s);
        write(sid,(void*)s,strlen(s));
        if(strlen(s)==0)
            break;
        sleep(1);
        read(sid,(void*)s1,sizeof(s1));
        printf("\n The received string is:%s\n",s1);
    }
    close(sid);
}

这里的代码只为一个字符进行通信。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2020-09-14
    • 2016-04-09
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-06-26
    相关资源
    最近更新 更多