地址查询函数的功能也就是通过主机名或者域名返回详细的主机信息,其中我们最常用的功能就是通过主机名获得主机的IP地址等信息。
接入阿里云IoT平台的硬件,首先要解析IoT的接入endpoint,每个产品接入域名都不同。规则是:
${productKey}.iot-as-mqtt.${regionId}.aliyuncs.com
getaddrinfo()函数
IPv6中引入了getaddrinfo()的新API,它是协议无关的,既可用于IPv4也可用于IPv6。getaddrinfo()函数能够处理名字到地址以及服务到端口这两种转换,返回的是一个addrinfo的结构(列表)指针而不是一个地址清单。
在此强烈推荐大家用getaddrinfo()函数替代已经过时的仅支持IPv4的gethostbyname()
函数原型
用主机名或服务名获取IP地址
头文件:,
int getaddrinfo(const char *restrict host,
const char *restrict service,
const struct addrinfo *restrict hints,
struct addrinfo **restrict result);
参数说明:
- host:一个主机名或者地址串(IPv4的点分十进制串或者IPv6的16进制串)
- service:服务名可以是十进制的端口号,也可以是已定义的服务名称,如ftp、http等
- hints:可以是一个空指针,也可以是一个指向某个addrinfo结构体的指针,调用者在这个结构中填入关于期望返回的信息类型的暗示。举例来说:指定的服务既可支持TCP也可支持UDP,所以调用者可以把hints结构中的ai_socktype成员设置成SOCK_DGRAM使得返回的仅仅是适用于数据报套接口的信息。
- result:本函数通过result指针参数返回一个指向addrinfo结构体链表的指针。
返回值:
- 若成功,返回0;若出错,返回非0错误码
addrinfo结构体:
struct addrinfo {
int ai_flags;
int ai_family;
int ai_socktype;
int ai_protocol;
socklen_t ai_addrlen; //ai_addr的地址长度
struct sockaddr *ai_addr;
char *ai_canonname;
struct addrinfo *ai_next; //指向链表的下一个结点
};
使用参考:
struct addrinfo *ai, *aip;
struct addrinfo hint;
struct sockaddr_in *sinp;
const char *addr;
int err;
char buf[1024];
hint.ai_flags = AI_CANONNAME;
hint.ai_family = 0;
hint.ai_socktype = 0;
hint.ai_protocol = 0;
hint.ai_addrlen = 0;
hint.ai_canonname = NULL;
hint.ai_addr = NULL;
hint.ai_next = NULL;
if((err = getaddrinfo("aliyun.com", NULL, &hint, &ai)) != 0)
printf("ERROR: getaddrinfo error: %s\n", gai_strerror(err));
for(aip = ai; aip != NULL; aip = aip->ai_next)
{
print_family(aip);
print_type(aip);
print_protocol(aip);
print_flags(aip);
printf("\n");
printf("Canonical Name: %s\n", aip->ai_canonname);
if(aip->ai_family == AF_INET)
{
sinp = (struct sockaddr_in *)aip->ai_addr;
addr = inet_ntop(AF_INET, &sinp->sin_addr, buf, sizeof buf);
printf("IP Address: %s ", addr);
printf("Port: %d\n", ntohs(sinp->sin_port));
}
printf("\n");
}