- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试创建一个简单的程序来获取给定特定主机名的 IP 地址:
我截取的代码附在下面:
#include <stdio.h>
#include <stdlib.h>
#include <stdio.h>
#include <netdb.h>
#include <sys/socket.h>
#include <errno.h>
#include <arpa/inet.h>
#include <string.h>
#include <unistd.h>
int main(int argc, char *argv[]) {
if(argc < 2){
printf("Please provide a hostname.\n");
exit(1);
}
char *hostname = argv[1];
char ip[100];
get_ip(hostname, ip);
printf("%s resolved to %s\n", hostname,ip);
}
int get_ip(char *hostname, char *ip){
struct sockaddr_in *h;
int sockfd;
struct addrinfo hints, *servinfo,*res;
struct addrinfo *iter;
int rv;
memset(&hints, 0, sizeof hints);
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
if((rv = getaddrinfo(hostname, "HTTP", &hints, &res)) != 0) {
fprintf(stderr, "getaddrinfo %s\n", gai_strerror(rv));
return 1;
}
for(iter=res; iter != NULL; iter=iter->ai_next){
printf("%p\n", iter->ai_next);
h=(struct sockaddr_in *)iter->ai_addr;
strcpy(IP, inet_ntoa(h->sin_addr));
printf("%s\n",ip);
}
freeaddrinfo(res);
return 0;
}
我输入以下参数:
gcc get_ip_addr.c -o get_ip_addr;
./get_ip_addr google-public-dns-b.google.com
这导致:
0x2475330
8.8.4.4
(nil)
0.0.0.0
当我删除 "http"
和 &hints
并将它们设置为 NULL
时,我得到以下结果:
0x1b63310
8.8.4.4
0x1b63360
8.8.4.4
0x1b633b0
8.8.4.4
0x1b63410
0.0.0.0
0x1b63470
0.0.0.0
(nil)
0.0.0.0
因此,当我将 service
和 hints
设置为 NULL
时,getaddrinfo()
返回多个可能的结果我不明白为什么我会得到多个 IP 地址而不是只得到一个 IP 地址?
最佳答案
通过将 service
和 hints
字段设置为 NULL
,您要求 getaddrinfo
返回所有可能的地址,其中包括 IP 和 IPv6,具有不同的套接字类型,您输出的只是 ip 地址的点表示,而这只是 getaddrinfo
返回的整个地址结构的一部分。如果仔细查看返回的地址,您会发现每个地址实际上是不同的。比如下面程序一共输出6个地址:
int main(int argc, char *argv[])
{
struct addrinfo *result, *rp;
int s;
s = getaddrinfo("google-public-dns-b.google.com", NULL, NULL, &result);
if (s != 0) {
fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(s));
exit(EXIT_FAILURE);
}
char addr[1024];
for (rp = result; rp != NULL; rp = rp->ai_next) {
printf("flags: 0x%x\tfamily: %d\tsocktype: %d\tprotocol: %d\n",
rp->ai_flags,
rp->ai_family,
rp->ai_socktype,
rp->ai_protocol);
s = getnameinfo(rp->ai_addr, rp->ai_addrlen, addr, sizeof addr, NULL, 0, NI_NUMERICHOST);
if (s != 0) {
printf("getnameinfo error:%d\n", s);
continue;
}
printf("addr: %s\n", addr);
}
freeaddrinfo(result);
return 0;
}
$ ./a.out
flags: 0x28 family: 2 socktype: 1 protocol: 6
addr: 8.8.4.4
flags: 0x28 family: 2 socktype: 2 protocol: 17
addr: 8.8.4.4
flags: 0x28 family: 2 socktype: 3 protocol: 0
addr: 8.8.4.4
flags: 0x28 family: 10 socktype: 1 protocol: 6
addr: 2001:4860:4860::8844
flags: 0x28 family: 10 socktype: 2 protocol: 17
addr: 2001:4860:4860::8844
flags: 0x28 family: 10 socktype: 3 protocol: 0
addr: 2001:4860:4860::8844
如您所见,返回的每个地址在地址、系列、套接字类型和协议(protocol)的组合方面都是不同的。如果您想返回一种特定类型的地址,请使用 hints
将返回的地址限制为您想要的地址。每个数据字段解释:
家庭:
#define PF_INET 2 /* IP protocol family. */
#define PF_INET6 10 /* IP version 6. */
#define AF_INET PF_INET
#define AF_INET6 PF_INET6
socks 类型:
enum __socket_type
{
SOCK_STREAM = 1, /* Sequenced, reliable, connection-based
byte streams. */
SOCK_DGRAM = 2, /* Connectionless, unreliable datagrams
of fixed maximum length. */
SOCK_RAW = 3, /* Raw protocol interface. */
....
}
协议(protocol):
enum {
IPPROTO_IP = 0, /* Dummy protocol for TCP */
IPPROTO_TCP = 6, /* Transmission Control Protocol */
IPPROTO_UDP = 17, /* User Datagram Protocol */
}
另一个问题,您在输出中看到“0.0.0.0”,因为您在 IPv6 地址上使用 inet_ntoa
而此函数仅支持 IPv4,因此它已被弃用,您应该考虑使用inet_ntop
或 getnameinfo
支持这两种地址族。
关于c - 为什么 getaddrinfo 有多个结果?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40782933/
我目前正在升级我们的软件以支持 ipv6,同时我正在将主机名/ip 字段扩展到最大主机名大小。在 sun 文档中,它似乎可以达到 1025(netdb.h:#define NI_MAXHOST 102
我正在使用 docker-compose 为开发人员提供技术栈 ROR、postgres、redis、mongo 的开发环境。 docker-compose build 运行成功,但是当我运行 doc
nc -zv cms-01 5061 失败: nc: getaddrinfo: 没有与主机名关联的地址 但是 ping/dig 工作正常。用getent可以看到最不一样的地方: # getent ho
我正在测试一条错误路径,该路径要求我删除来自 getaddrinfo 的请求。我设置了 2 个虚拟机: RHEL 7.9 Ubuntu 20 代码在两台机器上是相同的,只是为 test.com 调用
当我调用 getaddrinfo与 AF_UNSPEC ,理论上它可以返回任何它喜欢的地址族:AF_INET , AF_INET6 ,也许是 AppleTalk、蓝牙、datalink、netlink
我花了几个小时试图找出我的问题,只是在我写问题的时候找到了解决方案(当你需要正式化你的问题并解释它时,它总是有帮助的)。我发布它,希望它可以帮助某人。 使用 getaddrinfo,如果我尝试将套接字
我尝试通过给 getaddrinfo 赋予 http(s) 前缀来使用它,但它无法正确查找主机。我该如何解决这个问题。我要查 https://www.google.com但 gai_strerror(
我想在我的 header 中定义一个端口号(例如 #define port 9191 )并使用它来调用 getaddrinfo() 。但我收到错误,因为参数服务是 char const* 。如何使用#
我对 getaddrinfo() 的行为有疑问,这似乎取决于服务参数的指定方式。将服务指定为名称会导致 getaddrinfo() 使用 getservbyname() 来查询/etc/service
我有一个程序(用 C 语言编写),它使用 getaddrinfo() ,它的第一个参数是我希望它获取地址信息的主机名,假设它是“域”。在同一台计算机上的 /etc/hosts 文件中,我可以看到“do
int print_socket_info(int sock_fd, struct sockaddr_in *sin, short protocol){ char dbg[INET_ADDRS
这个问题已经存在: Login and Registration in Android with PHP, MySQL and SQLite 已关闭 7 年前。 我正在尝试连接到我的 mysql wa
我想连接sql数据库。我究竟做错了什么?请帮忙 常量.php 连接.php 最佳答案 首先,您的 constants.php 文件似乎在开始的 PHP 标记中有错字。您应该按如下方式更新文件:
我目前正在尝试编写自己的 HTTP 代理。我的代码可以正常工作,但我尝试使用 ApacheBench 对其进行压力测试,发现时不时地收到如下消息: “getaddrinfo:提供了节点名或服务名,或者
我正在使用 getaddrinfo() API 来解析跨平台 C++ 应用程序中的 DNS 查询。我需要知道 getaddrinfo() 在不同平台上的时间成本?谁能帮我解决这个问题? 最佳答案 这个
我有一个使用套接字的程序。在客户端,我通过 getaddrinfo 获取必要的信息,然后将结果传递给连接函数。事情是,它在不同的环境中工作不同。在获取 addrinfo 结构并将其传递给 connec
我有一个网络地址,例如 - 192.168.74.0/24。尝试使用 getaddrinfo 获取网络地址转换似乎不起作用。这里的名称是 192.168.74.0/24。 getaddrindo 的返
所以我花了一段时间才弄明白,因为“段错误”没有显示任何代码行,但显然这会导致我的程序出现段错误: if (getaddrinfo(options.servername, (char *)options
我遇到了一个奇怪的问题。我有这个 Python 应用程序,当我尝试在应用程序中打开 url 时,例如 urllib2.urlopen("http://google.com", None) 我收到以下错
我在 LAN 中用于某些套接字通信应用程序的目标机器具有用于 IPv4 的 192.168.98.91。 当我尝试使用 getaddrinfo() 解析 IP 地址时,它返回 127.0.0.1 如何
我是一名优秀的程序员,十分优秀!