设置简单UDP服务器时出现问题

时间:2012-07-13 18:40:11

标签: c sockets

我正在使用C进行套接字编程,但是我遇到了一个问题 - 我很确定 - 它来自我的困惑:如何使用 getaddrinfo()设置服务器,以及如何从客户端获取它的地址?我正在使用UDP。

(我来自Beej's tutorial,但我仍然不明白)

这是我服务器的代码(我试图在 mhgc.net 和端口3051运行它,虽然我还没有找到设置...主机名的方法?对不起,我很困惑):

int sockd;

/* Get my IP address */
struct addrinfo hints, * servinfo;
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_DGRAM;
hints.ai_flags = AI_PASSIVE;  // use my own address
if( getaddrinfo(NULL, "3051", &hints, &servinfo) != 0 ) {
    /* Yes, below is not... 'pretty code'; this is just an attempt */
    errorf("Failed to get server address: %s", gai_strerror( getaddrinfo(NULL, "3051", &hints, &servinfo) ));
    exit(EXIT_FAILURE);
    return 1;
}

struct addrinfo * p = NULL;
for(p = servinfo; p != NULL; p = p->ai_next) {
    if( (sockd = socket(p->ai_family, p->ai_socktype, p->ai_protocol)) == -1 ) {
        perror("Cannot open socket");
        continue;
    }

    if(bind(sockd, p->ai_addr, p->ai_addrlen) == -1) {
        close(sockd);
        perror("Cannot bind to port");
        continue;
    }

    break;
}

if(p == NULL) {
    error("listener: failed to bind socket");
    exit(EXIT_FAILURE);
    return 0;
}

freeaddrinfo(servinfo);

struct sockaddr_storage clientinfo;
socklen_t clientinfo_len = sizeof(clientinfo);

char buffer[15];
memset(buffer, 0, sizeof(buffer));
if( recvfrom(sockd, buffer, sizeof(buffer), 0, (struct sockaddr *)&clientinfo, &clientinfo_len) == -1 ) {
    perror("Error receiving packet");
    exit(EXIT_FAILURE);
    return 0;
}

char s[INET6_ADDRSTRLEN];
printf("listener: got packet from %s\n", inet_ntop(clientinfo.ss_family, get_in_addr((struct sockaddr *)&clientinfo), s, sizeof(s)));
printf("listener: packet contains \"%s\"\n", buffer);

/* Close the socket descriptor */
close(sockd);

    /* And now, exit, or whatever I need ... ... ... */

和我的客户:

int sockd;

struct addrinfo hints, * servinfo;
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_DGRAM;
//hints.ai_flags = AI_CANONNAME;
if( getaddrinfo(MKS_HOSTNAME, "www", &hints, &servinfo) != 0 ) {
    errorf("Failed to get server address: %s", gai_strerror( getaddrinfo(NULL, "3051", &hints, &servinfo) ));
    return 1;
}

struct addrinfo * p = NULL;
for(p = servinfo; p != NULL; p = p->ai_next) {
    if((sockd = socket(p->ai_family, p->ai_socktype, p->ai_protocol)) == -1) {
        perror("Cannot open socket");
        continue;
    }

    break;
}

if(p == NULL) {
    error("talker: failed to bind socket");
    return 2;
}

if( sendto(sockd, "this-is-a-test", strlen("this-is-a-test")+1, 0, p->ai_addr, p->ai_addrlen) == -1 ) {
    perror("Error sending packet");
    return 1;
}

freeaddrinfo(servinfo);

close(sockd);

错误 errorf 是帮助我调试应用程序的宏。别担心。

另一个提示,如果你允许我:来自Beej教程的talker.c,使用主机名“找到”服务器。我下载了listener.c和talker.c,编译了它(make listener&& make talker)但是,我不知道作为talker参数传递什么。监听器的主机名是什么?我该怎么找到它?再次抱歉。我真的很困惑:S

使用gcc编译器v4.4.5

使用Debian GNU / Linux“squeeze”

1 个答案:

答案 0 :(得分:0)

在服务器上,如果您不想收听特定地址,只需传递any地址(所有地址元素都设置为零)。您拨打getaddrinfo以获取本地地址的电话永远不会给您一个合理的答案。将service参数设置为NULL,因为您想知道主机的IP地址,而不是服务名称。

要将客户端连接到服务器,请使用服务器的IP地址(在服务器上执行/sbin/ifconfig并搜索客户端可以访问的地址)或其主机名(即执行{{ 1}}在服务器上并在客户端上使用该名称来连接服务器)。