c 中的 gethostbyname() 替代方案

gethostbyname() alternatives in c

我目前正在做一个网络游戏项目,我需要设计一个游戏,最多 3 个客户端可以连接到服务器,并且游戏在所有客户端和服务器之间进行。我在服务器端和客户端都使用 "sockaddr_in" 结构。

在我的游戏中,任何人都可以成为服务器,客户端需要提供正确的 IP 地址和端口号才能连接到服务器。当我分别在 "server_address.sin_addr.s_addr" 和 "server_address.sin_port" 中硬编码服务器 IP 地址和端口号的值时,游戏运行正常。但是硬编码不会解决我的任何人都是服务器并要求客户端输入服务器地址和端口号的问题。所以,我在客户端使用了 "gethostbyname()" 函数调用。但这并没有解决我的问题。 (原因可能是 gethostbyname() 在传递数字字符串时的行为未指定。(来源:link)。

以下是我在服务器端使用的代码:

struct sockaddr_in serv_addr, client_addr;
/* open a socket */
    if((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) 
    {
        err_ret = errno;
        return err_ret;
    }
    /* set initial values */
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons(PORT);
    serv_addr.sin_addr.s_addr = inet_addr(IP);
    memset(&(serv_addr.sin_zero), 0, 8);
    /* bind address with socket */
    if(bind(sockfd, (struct sockaddr *)&serv_addr, sizeof(struct sockaddr)) == -1) 
    {
        err_ret = errno;
        return err_ret;
    }

在客户端

struct sockaddr_in serv_addr;
struct hostent *to;
/* generate address */
if((to = gethostbyname(IP))==NULL) 
{
    err_ret = h_errno;
    fprintf(stderr, "gethostbyname() error...\n");
    return err_ret;
}
/* open a socket */
if((newfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) 
{
    err_ret = errno;
    fprintf(stderr, "socket() error...\n");
    return err_ret;
}
/* set initial values */
serv_addr.sin_family = AF_INET;
serv_addr.sin_port = htons(PORT);
memcpy(&serv_addr.sin_addr, to->h_addr_list[0], to->h_length);
memset(&(serv_addr.sin_zero), 0, 8);

这里有谁能告诉我一个有效的方法来执行上述过程吗?

如有任何帮助,我们将不胜感激。

getaddrinfo 已取代 gethostbyname。这应该可以更容易地从 IP 地址字符串创建 sockaddr_in 结构。

将数字形式的字符串或作为主机名转换为 sockaddr_in.

的示例代码
struct addrinfo hints = {};
addrinfo* pResultList = NULL;
struct sockaddr_in addr = {};
char* hostname = "1.2.3.4";

hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;
//  hints.ai_flags = AI_NUMERICHOST;  // if you know hostname is a numeric stirng, you can skip the DNS lookup by setting this flag

result = getaddrinfo(hostname , NULL, &hints, &pResultList);

if (result)
    memcpy(&addr, pResultList->ai_addr, sizeof(addr));

if (pResultList != NULL)
{
    ::freeaddrinfo(pResultList);
}