【问题标题】:UDP server with 2 UDP clients具有 2 个 UDP 客户端的 UDP 服务器
【发布时间】:2014-06-04 21:08:49
【问题描述】:

在我的代码中,客户端只向服务器发送数据。但是,我希望将 client1 所说的发送到服务器,然后服务器将其中继到 client2。我已经尝试了所有我能想到但无法弄清楚的方法。

服务器

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

int main() 
{ 
    int sock; 
    int addr_len, bytes_read; 
    char recv_data[1024]; 
    struct sockaddr_in server_addr , client_addr; 


    if ((sock = socket(AF_INET, SOCK_DGRAM, 0)) == -1) 
    { 
        perror("Socket"); 
        exit(1); 
    } 

    server_addr.sin_family = AF_INET; 
    server_addr.sin_port = htons(5000); 
    server_addr.sin_addr.s_addr = INADDR_ANY; 
    bzero(&(server_addr.sin_zero),8); 


    if (bind(sock,(struct sockaddr *)&server_addr, 
    sizeof(struct sockaddr)) == -1) 
    { 
        perror("Bind"); 
        exit(1); 
    } 

    addr_len = sizeof(struct sockaddr); 

    printf("\nUDPServer Waiting for client on port 5000"); 
    fflush(stdout); 

    while (1) 
    { 

        bytes_read = recvfrom(sock,recv_data,1024,0,(struct sockaddr *)&client_addr, &addr_len); 

        recv_data[bytes_read] = '\0'; 
        if(strlen(recv_data)>1)
        {
            printf("\n(%s , %d) said : ",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port)); 
            printf("%s", recv_data); 
            fflush(stdout); 
        }

        bytes_read = sendto(sock,recv_data,1024,0,(struct sockaddr *)&client_addr, &addr_len); 
        recv_data[bytes_read] = '\0';

    } 
 return 0; 
} 

客户

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

int main(int argc, char** argv) 
{ 
    int sock,i=0,bytes_read; 
    struct sockaddr_in server_addr; 
    struct hostent *host; 
    char send_data[1024],recv_data[1024];
    char nickname[20];
    if(argc != 2)  
    {  
        printf("Usage : %s <Server-IP>\n",argv[0]);  
        exit(0);  
    }  

    printf("Enter a nickname:");
    scanf("%s",&nickname);

    printf("WELCOME %s ..Please (Enter) to start chat or Quit(q): ",&nickname);
    host= (struct hostent *) gethostbyname((char *)argv[1]); 


    if ((sock = socket(AF_INET, SOCK_DGRAM, 0)) == -1) 
    { 
        perror("socket"); 
        exit(1); 
    } 

    server_addr.sin_family = AF_INET; 
    server_addr.sin_port = htons(5000); 
    server_addr.sin_addr = inet_addr(argv[1]); //IP ADDRESS 
    bzero(&(server_addr.sin_zero),8); 


    while (1) 
    { 
        if(i>1)
        {
            printf("%s:",&nickname); 
        }
        gets(send_data); 

        if ((strcmp(send_data , "q") == 0) || strcmp(send_data , "Q") == 0) 
            break; 
        else
        {
            sendto(sock, send_data, strlen(send_data), 0,(struct sockaddr *)&server_addr, sizeof(struct sockaddr)); 


        bytes_read = recvfrom(sock, send_data, strlen(send_data), 0,(struct sockaddr *)&server_addr, sizeof(struct sockaddr)); 
        recv_data[bytes_read] = '\0';
        }
    i++;
    } 
} 

【问题讨论】:

  • 您有bytes_read,其中包含读取的字节数。那你为什么打电话给strlen?另外,您是否意识到您将其发送回您收到它的同一个套接字?
  • 我的主要问题是一个服务器的多个客户端。我不向客户端发送服务器数据。我该怎么做?
  • 一个服务器通常用于管理多个客户端连接。您的代码只是管理一个连接。您应该阐明服务器/客户端模型的工作原理。我建议阅读这个优秀的在线教程:beej.us/guide/bgnet
  • 我不应该说“从同一个套接字”,我应该说“到您收到它的目的地”。
  • 还要注意 bytes_read 可以是 1024,所以recv_data[bytes_read] = '\0'; 可以写入超出缓冲区的大小。

标签: c linux sockets udp udpclient


【解决方案1】:

现在,您的服务器正在将收到的消息发送回发送该消息的同一客户端。要执行您的要求,您需要保留已向您的服务器发送消息的所有客户端的列表。

当您收到来自未知客户端的消息时,将该客户端添加到列表中。如果您在一段时间内没有收到来自该客户端的新消息,请将其从列表中删除。如果您需要让客户端长时间保持“连接”到服务器,客户端必须定期向服务器发送 keepalive/heartbeat 消息以避免超时。

每次收到需要中继的消息时,您可以循环遍历列表,根据需要将消息发送给每个客户端。

例如(伪代码,我将把实现留作练习让你弄清楚):

int main() 
{ 
    CreateAndBindServerSocket();

    while (1) 
    { 
        if (ReadAMessage(&msg, &sender))
        {
            if (!ClientIsInList(sender))
                AddClientToListAndStartIdleTimer(sender);
            else
                RestartIdleTimerForClient(sender);

            if (ShouldRelayMessageToAClient(msg, &recip))
            {
                if (FindClientInList(recip, &recip_addr))
                {
                    SendMsgToClient(recip_addr, msg);
                }
            }
            else if (ShouldRelayMessageToAllClients(msg))
            {
                for (each recip_addr in list)
                {
                    SendMsgToClient(recip_addr, msg)
                }
            }
        }

        RemoveTimedOutClientsFromList();
    } 

    return 0; 
} 

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2018-01-04
    • 1970-01-01
    • 1970-01-01
    • 2020-06-22
    • 2012-01-04
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多