【问题标题】:multithread server/client implementation in CC中的多线程服务器/客户端实现
【发布时间】:2026-01-17 21:00:01
【问题描述】:

我刚刚开始学习基本的网络概念。我正在尝试在 C 中实现多线程服务器-客户端程序。但问题是不是为客户端运行多个窗口/终端/实例,我应该使用 fork() 来创建client.so 通过创建客户端的子级,将创建多个客户端。现在这些子客户端中的每一个都将在线程上与服务器通信。

之前我创建了一个类似的程序,但是对于多个客户端,您必须为客户端打开多个窗口并运行所有窗口。

我在哪里修改我的代码时遇到了麻烦(在服务器和客户端代码中。我认为服务器一是好的。但我不知道在客户端程序中的哪个位置 fork() 以及应该进行哪些更改)。

实际上我不想打开多个窗口来运行多个客户端,这就是为什么我使用 fork() 创建它的多个副本。有没有其他方法可以创建多个客户端并将它们连接到我的通过线程进行服务器编程。

服务器:

// socket server example, handles multiple clients using threads

#include<stdio.h>
#include<string.h>    //strlen
#include<stdlib.h>    //strlen
#include<sys/socket.h>
#include<arpa/inet.h> //inet_addr
#include<unistd.h>    //write
#include<pthread.h> //for threading , link with lpthread

//the thread function
void *connection_handler(void *);

int main(int argc , char *argv[])
{
    int socket_desc , client_sock , c , *new_sock;
    struct sockaddr_in server , client;

    //Create socket
    socket_desc = socket(AF_INET , SOCK_STREAM , 0);
    if (socket_desc == -1)
    {
        printf("Could not create socket");
    }
    puts("Socket created");

    //Prepare the sockaddr_in structure
    server.sin_family = AF_INET;
    server.sin_addr.s_addr = INADDR_ANY;
    server.sin_port = htons( 3000 );

    //Bind
    if( bind(socket_desc,(struct sockaddr *)&server , sizeof(server)) < 0)
    {
        //print the error message
        perror("bind failed. Error");
        return 1;
    }
    puts("bind done");

    //Listen
    listen(socket_desc , 3);

    //Accept and incoming connection
    puts("Waiting for incoming connections...");
    c = sizeof(struct sockaddr_in);

        c=sizeof(struct sockaddr_in);
       while(client_sock=accept(socket_desc,(struct sockaddr*)&client,(socklen_t*)&c))
       {
        puts("Connection accepted");

        pthread_t sniffer_thread;
        new_sock = malloc(1);
        *new_sock = client_sock;

        if( pthread_create( &sniffer_thread , NULL ,  connection_handler , (void*) new_sock) < 0)
        {
            perror("could not create thread");
            return 1;
        }

        puts("Handler assigned");
    }

    if (client_sock < 0)
    {
        perror("accept failed");
        return 1;
    }
    return 0;
}
/*
  This will handle connection for each client
  */
void *connection_handler(void *socket_desc)
{
    //Get the socket descriptor
    int sock = *(int*)socket_desc;
    int n;

        char    sendBuff[100], client_message[2000];

      while((n=recv(sock,client_message,2000,0))>0)
      {

        send(sock,client_message,n,0);
      }
      close(sock);

      if(n==0)
      {
        puts("Client Disconnected");
      }
      else
      {
        perror("recv failed");
      }
    return 0;
}

客户:

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

#define MAX_SIZE 50

int main()
{
    int sock_desc;
    struct sockaddr_in serv_addr;
    char sbuff[MAX_SIZE],rbuff[MAX_SIZE];

    if((sock_desc = socket(AF_INET, SOCK_STREAM, 0)) < 0)
        printf("Failed creating socket\n");

    bzero((char *) &serv_addr, sizeof (serv_addr));

    serv_addr.sin_family = AF_INET;
    serv_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
    serv_addr.sin_port = htons(3000);

    if (connect(sock_desc, (struct sockaddr *) &serv_addr, sizeof (serv_addr)) < 0) {
        printf("Failed to connect to server\n");
        return -1;
    }

    printf("Connected successfully - Please enter string\n");
    while(fgets(sbuff, MAX_SIZE , stdin)!=NULL)
    {
      send(sock_desc,sbuff,strlen(sbuff),0);

          if(recv(sock_desc,rbuff,MAX_SIZE,0)==0)
           printf("Error");
          else
           fputs(rbuff,stdout);

       bzero(rbuff,MAX_SIZE);//to clean buffer-->IMP otherwise previous word characters also came
    }
        close(sock_desc);
    return 0;

}

【问题讨论】:

  • 为什么要在客户端fork()?这很不寻常
  • @Felipe:如果客户端是测试代码来模拟连接到服务器的多个客户端,这将是有意义的。
  • 如果你想学习网络编程,我建议你看看比 fork 更现代的概念。调查基于线程或事件的 IO。分叉是网络程序的原始且过时的模型。
  • Filipe:这就是我的老师要求我做的。实际上我不想打开多个窗口来运行多个客户端,这就是为什么我使用 fork() 创建它的多个副本。是还有其他方法可以让我创建多个客户端并通过线程将它们连接到我的服务器 prog。
  • 你忘记在 connection_handler 中释放 (socket_desc)。

标签: c multithreading tcp operating-system


【解决方案1】:

您可以使用线程创建多个客户端。为每个客户端创建一个单独的线程,然后从线程处理程序连接到服务器。我不确定这是否是一个好方法。

代码:

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

#define MAX_SIZE 50
#define NUM_CLIENT 5
void *connection_handler(void *socket_desc);
int main()
{
    int socket_desc , new_socket , c , *new_sock, i;
    pthread_t sniffer_thread;
    for (i=1; i<=NUM_CLIENT; i++) {
        if( pthread_create( &sniffer_thread , NULL ,  connection_handler , (void*) i) < 0)
        {
            perror("could not create thread");
            return 1;
        }
        sleep(3);
    }
    pthread_exit(NULL);
    return 0;
}

void *connection_handler(void *threadid)
{
    int threadnum = (int)threadid;
    int sock_desc;
    struct sockaddr_in serv_addr;
    char sbuff[MAX_SIZE],rbuff[MAX_SIZE];

    if((sock_desc = socket(AF_INET, SOCK_STREAM, 0)) < 0)
        printf("Failed creating socket\n");

    bzero((char *) &serv_addr, sizeof (serv_addr));

    serv_addr.sin_family = AF_INET;
    serv_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
    serv_addr.sin_port = htons(8888);

    if (connect(sock_desc, (struct sockaddr *) &serv_addr, sizeof (serv_addr)) < 0) {
        printf("Failed to connect to server\n");
    }

    printf("Connected successfully client:%d\n", threadnum);
    while(1)
    {
        printf("For thread : %d\n", threadnum);
        fgets(sbuff, MAX_SIZE , stdin);
        send(sock_desc,sbuff,strlen(sbuff),0);

        if(recv(sock_desc,rbuff,MAX_SIZE,0)==0)
            printf("Error");
        else
           fputs(rbuff,stdout);

        bzero(rbuff,MAX_SIZE);
        sleep(2);
    }
    close(sock_desc);
    return 0;
}

为了便于理解,我使用了sleep

参考:

http://www.amazon.com/UNIX-Network-Programming-Richard-Stevens/dp/0139498761

http://beej.us/guide/bgnet/

https://computing.llnl.gov/tutorials/pthreads/

【讨论】:

  • sujin:收到以下警告和错误 mthclient.c:在函数“main”中:mthclient.c:18:警告:从不同大小的整数 mthclient.c 转换为指针:在函数“connection_handler”中:mthclient.c:31:警告:从指针转换为不同大小的整数/tmp/ccuPwTxp.o:在函数main'中:mthclient.c:(.text + 0x2d):未定义对pthread_create的引用'collect2:ld返回1退出状态 lnxrc186055-14:/shikhar # ./c -bash: ./c: 没有这样的文件或目录
  • 它工作正常。但有些同步。我认为有问题....服务器:-套接字创建绑定完成等待传入连接...连接接受处理程序分配连接接受处理程序分配连接接受处理程序分配连接接受处理程序分配连接接受处理程序分配客户端断开客户端断开客户端断开客户端断开连接的客户端断开连接
  • 客户端:-连接成功客户端:1 对于线程:1 hello hello $ 连接成功客户端:2 对于线程:2 hi ads $ 连接成功客户端:3 对于线程:3 ads adas asd $Connected成功客户端:4 对于线程:4 adas $Connected 成功客户端:5 对于线程:5 asd........我无法以正确的格式发布输出。抱歉......'hi'我写道,当客户端 1 连接时会在客户端 2 中回显。为什么?????
  • sujin:你能看懂我贴的o/p吗,还是把o/ps的截图发给你。我不知道怎么在*这里发截图。跨度>
  • 所有线程共享同一个终端,所以可能会有冲突。我真的不明白这个应用程序有什么需要,只是一个家庭作业吗?
【解决方案2】:

首先,如果您fork(),您将创建额外的进程,而不是额外的线程。要创建其他线程,您需要使用pthread_create

其次,作为一名学生,这里的标准答案是“阅读斯蒂芬斯”。即使对于我们这些在编写套接字 I/O 例程方面经验丰富的人来说,这不仅是一个非常宝贵的工具,而且它还包含非线程非分叉异步 I/O 的示例,以及向它们添加线程和分叉的各种方法。我相信你想要的是:http://www.amazon.com/Programming-Environment-Addison-Wesley-Professional-Computing/dp/0321637739(如果没记错的话,第 14 章)。这应该在你的大学图书馆里。

【讨论】:

最近更新 更多