【问题标题】:Whats wrong with my UDP Client Server?我的 UDP 客户端服务器出了什么问题?
【发布时间】:2009-12-01 23:23:07
【问题描述】:

我有一个 UDP 服务器,它应该等待客户端连接到它,然后向它发送一个文件名字符串。现在我只想让它将文件名回显给客户端。这是我的代码

服务器

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

#define MAXBUFLEN 1024

// Usage: ./server PORT
int main(int argc, char *argv[])
{
    int sockfd;
    struct sockaddr_in server;
    size_t clientSize;
    struct sockaddr_storage client;
    char buf[MAXBUFLEN];
    int portno = atoi(argv[1]);
    int numbytes;

    printf("Port: %d\n", portno);

    // Create UDP Socket
    if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) == -1) {
        perror("Can't create socket");
        exit(-1);
    }

    // Configure socket
    memset(&server, 0, sizeof server);
    server.sin_family = AF_INET; // Use IPv4
    server.sin_addr.s_addr = INADDR_ANY; // My IP
    server.sin_port = htons(portno); // Server Port

    // Bind socket
    if ((bind(sockfd, (struct sockaddr *) &server, sizeof(server))) == -1) {
        close(sockfd);
        perror("Can't bind");
    }

    while (1) {
        printf("Waiting for data...\n");

        // Receive data from Client
        clientSize = sizeof(client);
        numbytes = recvfrom(sockfd, buf, MAXBUFLEN-1,0,
                (struct sockaddr *) &client, &clientSize);

        buf[numbytes] = '\0';

        printf("client sent: %s\n", buf);

        // Rely to client
        sendto(sockfd, buf, MAXBUFLEN-1, 0,
                (struct sockaddr *) &client, &clientSize);
    }

    printf("Closing");
    close(sockfd);

    return 0;
}

客户

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

#define MAXBUFLEN 1024

//Usage: ./client PORT IP FILE
int main(int argc, char *argv[]) {
    int sockfd;
    struct sockaddr_in server;
//  struct sockaddr_storage client;
    char buf[MAXBUFLEN];
    int portno = atoi(argv[1]);
    char *serverIP = argv[2];
    char *filename = argv[3];
    int numbytes;

    printf("Port: %d, IP:%s, File:%s\n", portno, serverIP, filename);

    // Create UDP Socket
    if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) == -1) {
        perror("Can't create socket");
        exit(-1);
    }

    // The server IP and Port
    memset(&server, 0, sizeof server);
    server.sin_family = AF_INET; // Use IPv4
    server.sin_port = htons(portno); // Server Port
    struct hostent *hp = gethostbyname(serverIP);
    memcpy((char*)&server.sin_addr, (char*)hp->h_name, hp->h_length);

    printf("Sent %s\n", filename);

    // No need to bind, just send request for file
    int serverSize = sizeof(server);
    sendto(sockfd, argv[3], strlen(argv[3]), 0,
            (struct sockaddr *) &server, serverSize);

    printf("Waiting for reply\n");

    recvfrom(sockfd, buf, MAXBUFLEN-1, 0,
            (struct sockaddr *) &server, serverSize);

    buf[numbytes] = '\0';

    printf("server sent: %s\n", buf);

    close(sockfd);
}

到目前为止,服务器只是打印出来

Waiting for data...

只有一次......所以它永远不会收到请求。

客户端打印出来:

./client 8083 127.0.0.1 hello
Port: 8083, IP:127.0.0.1, File:hello
Sent hello
Waiting for reply
server sent: 

请帮忙,我是 C 和 UDP 套接字世界的新手。

哦,是的,我已经阅读了 Beej 的指南......请不要只是将其发布为答案。

【问题讨论】:

    标签: c sockets udp


    【解决方案1】:

    首先,在调用您的服务器的 sendto 时存在错误。 clientSize 应该是一个值,而不是指针。另外,正如 Tom 所说,应该将 numbytes 设置为 recvfrom 的结果。

    最大的问题可能是您将 hp->h_name 复制到服务器的 sin_addr 元素中。 h_name 是主机的 name,而不是 IP 地址。换个试试

    
        memcpy((char*)&server.sin_addr, (char*)hp->h_name, hp->h_length);
    

    
        memcpy((char*)&server.sin_addr, (char*)hp->h_addr_list[0], hp->h_length);
    

    在您的客户端中。

    【讨论】:

    • 太棒了!非常感谢。我最终改变了我的代码。我将发布我现在正在运行的答案。
    【解决方案2】:

    您的客户端中的 numbytes 未初始化,也未使用!再次检查您的代码....

    希望这会有所帮助, 最好的祝福, 汤姆。

    【讨论】:

      【解决方案3】:

      修复了错误!谢谢你,汤米。这是我现在正在运行的内容。

      服务器:

      #include <sys/types.h>
      #include <sys/socket.h>
      #include <netinet/in.h>
      #include <netdb.h>
      #include <stdio.h>
      
      #define MAXBUFLEN 1024
      
      void error(char *msg) {
          perror(msg);
          exit(0);
      }
      
      int main(int argc, char *argv[]) {
          int sockfd;
          struct sockaddr_in server, client;
          int serverSize, clientSize, numbytes;
          char buf[MAXBUFLEN];
          int portno = atoi(argv[1]);
      
          if (argc < 2) {
              fprintf(stderr, "ERROR, no port provided\n");
              exit(0);
          }
      
          printf("Running on Port: %d\n", portno);
      
          // Create UDP Socket
          sockfd = socket(AF_INET, SOCK_DGRAM, 0);
          if (sockfd < 0) {
              error("Can't create socket");
          }
      
          // Configure Server Info
          serverSize = sizeof(server);
          bzero(&server, serverSize);
          server.sin_family = AF_INET;
          server.sin_addr.s_addr = INADDR_ANY;
          server.sin_port = htons(portno);
      
          // Bind
          if (bind(sockfd, (struct sockaddr *) &server, serverSize) < 0) {
              error("binding");
          }
      
          clientSize = sizeof(struct sockaddr_in);
      
          while (1) {
              printf("Waiting for Client Request...\n");
              numbytes = recvfrom(sockfd, buf, MAXBUFLEN, 0,
                      (struct sockaddr *) &client, &clientSize);
              if (numbytes < 0) {
                  error("recvfrom");
              }
              buf[numbytes] = '\0';
      
              printf("Received request for File: %s\n", buf);
              numbytes = sendto(sockfd, buf, sizeof(buf), 0,
                      (struct sockaddr *) &client, clientSize);
              if (numbytes < 0) {
                  error("sendto");
              }
              sendFile(sockfd, client);
          }
      }
      
      sendFile(int sockfd, struct sockaddr_in client) {
      
      }
      

      客户:

      #include <sys/types.h>
      #include <sys/socket.h>
      #include <netinet/in.h>
      #include <arpa/inet.h>
      #include <netdb.h>
      #include <stdio.h>
      
      #define MAXBUFLEN 1024
      
      void error(char *msg) {
          perror(msg);
          exit(0);
      }
      
      int main(int argc, char *argv[]) {
          int sockfd;
          struct sockaddr_in server, client;
          int serverSize;
          struct hostent *hp;
          char buf[MAXBUFLEN];
          int numbytes;
          char *serverIP = argv[1];
          int portno = atoi(argv[2]);
          char *filename = argv[3];
          int done = 1;
      
          if (argc != 4) {
              printf("Usage: SERVER PORT FILENAME\n");
              exit(1);
          }
      
          // Create Socket
          sockfd = socket(AF_INET, SOCK_DGRAM, 0);
          if (sockfd < 0) {
              error("Can't create socket");
          }
      
          // Configure Server Info
          server.sin_family = AF_INET;
          hp = gethostbyname(serverIP);
          if (hp == 0)
              error("Unknown host");
          bcopy((char *) hp->h_addr, (char *) &server.sin_addr, hp->h_length);
          server.sin_port = htons(portno);
          serverSize = sizeof(struct sockaddr_in);
      
          // Send Message
          strcat(buf, filename);
          numbytes = sendto(sockfd, buf, strlen(buf), 0, &server, serverSize);
          if (numbytes < 0)
              error("Sendto");
          bzero(buf, MAXBUFLEN);
      
          // Build file then set Done = 0
          while (done) {
              numbytes = recvfrom(sockfd, buf, MAXBUFLEN, 0, &client, &serverSize);
              if (numbytes < 0)
                  error("recvfrom");
      
              done = 0;
          }
      
          printf("Received the packet: %s\n", buf);
      }
      

      我还在这里找到了一个很好的例子:http://www.linuxhowtos.org/C_C++/socket.htm

      感谢大家的帮助!

      【讨论】:

      • 顺便说一句,b* 函数(bzero()、bcopy 等)是“遗留”BSD 库调用; C 标准喜欢 mem* 版本。哦,而且 struct hostent 中的 h_addr 元素实际上是 h_addr_lest[0] 的宏,以防将来混淆。
      【解决方案4】:

      愚蠢的问题,但是您的防火墙是否阻止了端口 8083。

      鲍勃。

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 2011-08-29
        • 1970-01-01
        • 2020-09-30
        • 2020-06-22
        • 2012-01-04
        相关资源
        最近更新 更多