getaddrinfo函数是一个用于解析主机名和服务名的网络地址信息的函数。它的原型如下:
int getaddrinfo(const char *node, const char *service, const struct addrinfo *hints, struct addrinfo **res);
该函数根据传入的参数,从系统的网络配置中获取与指定主机名(或IP地址)和服务名相匹配的网络地址结构列表。
参数说明:
node
:要解析的主机名或IP地址字符串。service
:要解析的服务名或端口号字符串。hints
:提供了一些可选参数,用于限制查询结果以及设置其他标志位等。res
:指向一个addrinfo结构链表的指针,函数将会把查询结果存储在这个链表中。
返回值为0表示函数执行成功,否则返回非零错误码来表示错误类型。
以下是对getaddrinfo函数使用过程的一般步骤:
- 创建并初始化一个addrinfo结构体作为hints参数,并设置相关标志位和限制条件。可以通过memset或者直接赋值进行初始化。
- 调用getaddrinfo函数,传入node、service、hints和res参数。
- 检查返回值是否为0。如果不是0,则发生了错误,可以通过调用gai_strerror函数打印具体错误信息。
- 遍历返回的addrinfo链表(以res为头指针),逐个访问其中包含的每个地址信息。
以下是一个简单示例代码,演示了如何使用getaddrinfo函数来解析主机名和服务名,并打印出返回的地址信息:
#include <stdio.h>
#include <stdlib.h>
#include <netdb.h>
int main() {
const char *host = "www.example.com";
const char *port = "80";
struct addrinfo hints, *res;
// 初始化hints结构体
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
// 调用getaddrinfo函数进行解析
int status = getaddrinfo(host, port, &hints, &res);
if (status != 0) {
fprintf(stderr, "getaddrinfo error: %s\n", gai_strerror(status));
exit(1);
}
// 遍历并打印结果链表中的每个地址信息
struct addrinfo *p;
for (p = res; p != NULL; p = p->ai_next) {
char ip[INET6_ADDRSTRLEN];
void *addr;
if (p->ai_family == AF_INET) { // IPv4
struct sockaddr_in *ipv4 = (struct sockaddr_in *)p->ai_addr;
addr = &(ipv4->sin_addr);
} else { // IPv6
struct sockaddr_in6 *ipv6 = (struct sockaddr_in6 *)p->ai_addr;
addr = &(ipv6->sin6_addr);
}
inet_ntop(p->ai_family, addr, ip, sizeof(ip));
printf("IP address: %s\n", ip);
}
freeaddrinfo(res); // 释放链表
return 0;
}
请注意,在实际使用getaddrinfo函数时,您可能需要根据具体需求设置hints结构体中的参数,并对返回的地址信息进行适当的处理和选择。此外,还需要在程序结束后调用freeaddrinfo函数来释放返回的链表内存。
内容由零声教学AI助手提供,问题来源于学员提问