【问题标题】:How to Test and Run a Client Server Program in C++ using Eclipse如何使用 Eclipse 在 C++ 中测试和运行客户端服务器程序
【发布时间】:2019-10-12 14:17:14
【问题描述】:

我正在尝试开始使用 UDP,并想测试和调试一些客户端服务器程序。

我正在使用带有 cygwin64 作为编译器的 Eclipse IDE。

我从这里找到了一些示例客户端服务器程序:https://www.geeksforgeeks.org/udp-server-client-implementation-c/

我希望能够运行该示例,让我开始了解 UDP 客户端服务器。

示例代码如下:

服务器

    // Server side implementation of UDP client-server model
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>

#define PORT     8080
#define MAXLINE 1024

// Driver code
int main() {
    int sockfd;
    char buffer[MAXLINE];
    char *hello = "Hello from server";
    struct sockaddr_in servaddr, cliaddr;

    // Creating socket file descriptor
    if ( (sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0 ) {
        perror("socket creation failed");
        exit(EXIT_FAILURE);
    }

    memset(&servaddr, 0, sizeof(servaddr));
    memset(&cliaddr, 0, sizeof(cliaddr));

    // Filling server information
    servaddr.sin_family    = AF_INET; // IPv4
    servaddr.sin_addr.s_addr = INADDR_ANY;
    servaddr.sin_port = htons(PORT);

    // Bind the socket with the server address
    if ( bind(sockfd, (const struct sockaddr *)&servaddr,
            sizeof(servaddr)) < 0 )
    {
        perror("bind failed");
        exit(EXIT_FAILURE);
    }



    int len, n;
    n = recvfrom(sockfd, (char *)buffer, MAXLINE,
                MSG_WAITALL, ( struct sockaddr *) &cliaddr,
                &len);
    buffer[n] = '\0';
    printf("Client : %s\n", buffer);
    sendto(sockfd, (const char *)hello, strlen(hello),
        0, (const struct sockaddr *) &cliaddr,
            len);
    printf("Hello message sent.\n");

    return 0;
}

客户

// Client side implementation of UDP client-server model 
#include <stdio.h> 
#include <stdlib.h> 
#include <unistd.h> 
#include <string.h> 
#include <sys/types.h> 
#include <sys/socket.h> 
#include <arpa/inet.h> 
#include <netinet/in.h> 

#define PORT     8080 
#define MAXLINE 1024 

// Driver code 
int main() { 
    int sockfd; 
    char buffer[MAXLINE]; 
    char *hello = "Hello from client"; 
    struct sockaddr_in     servaddr; 

    // Creating socket file descriptor 
    if ( (sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0 ) { 
        perror("socket creation failed"); 
        exit(EXIT_FAILURE); 
    } 

    memset(&servaddr, 0, sizeof(servaddr)); 

    // Filling server information 
    servaddr.sin_family = AF_INET; 
    servaddr.sin_port = htons(PORT); 
    servaddr.sin_addr.s_addr = INADDR_ANY; 

    int n, len; 

    sendto(sockfd, (const char *)hello, strlen(hello), 
        MSG_CONFIRM, (const struct sockaddr *) &servaddr,  
            sizeof(servaddr)); 
    printf("Hello message sent.\n"); 

    n = recvfrom(sockfd, (char *)buffer, MAXLINE,  
                MSG_WAITALL, (struct sockaddr *) &servaddr, 
                &len); 
    buffer[n] = '\0'; 
    printf("Server : %s\n", buffer); 

    close(sockfd); 
    return 0; 
} 

我已尝试打开两个不同的 Eclipse 工作台并运行这两个代码,但它没有按预期运行,它表示消息已发送但我无法在客户端或服务器上接收它们。

如果可能的话,我绝对想坚持使用 c/c++ 并尝试在 eclipse 中使用它。

如果有人对我如何能够看到一些结果或任何可以让我开始做的事情有任何建议,我将不胜感激。谢谢!

【问题讨论】:

  • 无关:我还没有看到为 UDP 实现 MSG_WAITALL,因为它是面向数据包的。你要么得到一个数据包,要么你没有。如果您不读取整个数据包,则缓冲区太小,那是您的问题。如果您的消息超过了数据包的最大大小,则消息将被分片,如果数据包丢失或发送无序,则由您自行清理。

标签: c++ eclipse server udp client-server


【解决方案1】:

问题

您在客户端代码中没有正确设置服务器的地址。

servaddr.sin_addr.s_addr = INADDR_ANY;

发送时无意义(为什么要发送到任何可用地址?您要发送到服务器。)并被sendto拒绝。如果您检查了返回值,您会看到这一点。始终检查返回值,即使是 UDP 通信。当然,在sendto 时您无法检测到数据包无法通过 UDP 到达的原因有上百万个,但很高兴知道网络堆栈实际上接受了该消息,即使 grue 随后吞噬在互联网的黑暗角落之一的数据包。一个发送错误,你可以做一些事情,比如perror,然后相应地修复代码。一个grue,不是那么多。您必须使协议足够健壮以重新传输或以其他方式在数据包丢失后幸存下来。

解决方案

getaddrinfo获取服务器的地址结构。

替换

// Filling server information 
servaddr.sin_family = AF_INET; 
servaddr.sin_port = htons(PORT); 
servaddr.sin_addr.s_addr = INADDR_ANY; 

类似的东西

struct addrinfo hints;
struct addrinfo *hostlist;
memset(&hints, 0, sizeof(struct addrinfo));
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_DGRAM;
hints.ai_flags = 0;
hints.ai_protocol = IPPROTO_UDP; 

if (getaddrinfo("Server Name or Address Goes Here",
                PORT, // this needs to be a char *, not an integer
                &hints,
                &hostlist))
{
    // handle error
    exit(EXIT_FAILURE);
}

PORT 在这种情况下需要是char *,所以不是

#define PORT     8080 

使用

constexpr char const * PORT = "8080";

如果您的编译器较旧且不支持,您可以省略 constexpr

然后当你去发送时,遍历hostlist,直到你找到一个响应的主机。由于您可能已经将服务器范围缩小到一个候选人,这可能是矫枉过正,但您不妨开始练习正确地做事。当系统开始变得复杂时,替代方案很糟糕。

struct addrinfo *curhost;
for (curhost = hostlist; curhost != NULL; curhost = curhost->ai_next)
{
    int rval = sendto(sockfd,
                      (const char *) hello,
                      strlen(hello),
                      0,
                      curhost->ai_addr,
                      curhost->ai_addrlen);
    if (rval> 0) // always check return codes. Programmers are lazy. 
                 // They wouldn't have gone to the effort of putting
                 // it there if it wasn't important.
    {
        if server responds
            do protocol stuff to complete transaction
            break;
    }
}
freeaddrinfo(hostlist); // thou shalt not leak resources.
if (curhost == nullptr)
{
    notify user that no server was willing to talk
}

Documentation for getaddrinfo.

【讨论】:

  • 各个层面的反应都很好。要回答 OP 的原始问题:“是的,在单独的工作台中的两个 Eclipse 实例是同时运行两个程序的好方法。” @Darklink9110 - 修复 user4581301 指出的编码错误,你应该很高兴。另外:查看Beej's Guide to Network Programming
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2015-08-22
  • 2010-09-07
  • 1970-01-01
  • 2016-10-03
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多