Skip to content

Latest commit

 

History

History
135 lines (93 loc) · 4.91 KB

56.md

File metadata and controls

135 lines (93 loc) · 4.91 KB

网络,第 2 部分:使用 getaddrinfo

原文:https://github.com/angrave/SystemProgramming/wiki/Networking%2C-Part-2%3A-Using-getaddrinfo

如何使用getaddrinfo将主机名转换为 IP 地址?

函数getaddrinfo可以将人类可读域名(例如www.illinois.edu)转换为 IPv4 和 IPv6 地址。实际上它将返回 addrinfo 结构的链表:

struct addrinfo {
    int              ai_flags;
    int              ai_family;
    int              ai_socktype;
    int              ai_protocol;
    socklen_t        ai_addrlen;
    struct sockaddr *ai_addr;
    char            *ai_canonname;
    struct addrinfo *ai_next;
};

它非常易于使用。例如,假设您想在 www.bbc.com 中找到网络服务器的数字 IPv4 地址。我们分两个阶段完成。首先使用 getaddrinfo 构建可能的连接的链表。其次使用getnameinfo将二进制地址转换为可读形式。

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>

struct addrinfo hints, *infoptr; // So no need to use memset global variables

int main() {
  hints.ai_family = AF_INET; // AF_INET means IPv4 only addresses

  int result = getaddrinfo("www.bbc.com", NULL, &hints, &infoptr);
  if (result) {
    fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(result));
    exit(1);
  }

  struct addrinfo *p;
  char host[256],service[256];

  for(p = infoptr; p != NULL; p = p->ai_next) {

    getnameinfo(p->ai_addr, p->ai_addrlen, host, sizeof(host), service, sizeof(service), NI_NUMERICHOST);
    puts(host);
  }

  freeaddrinfo(infoptr);
  return 0;
}

典型输出:

212.58.244.70
212.58.244.71 

www.cs.illinois.edu 如何转换为 IP 地址?

魔法!不用说,使用称为“DNS”(域名服务)的系统。如果计算机没有在本地保存答案,则它会将 UDP 数据包发送到本地 DNS 服务器。该服务器又可以查询其他上游 DNS 服务器。

DNS 安全吗?

DNS 本身很快但不安全。 DNS 请求未加密,容易受到“中间人”攻击。例如,咖啡店互联网连接可以轻易地破坏您的 DNS 请求并发回特定域的不同 IP 地址

如何连接到 TCP 服务器(例如 Web 服务器?)

TODO 连接到远程计算机需要三个基本系统调用:

getaddrinfo -- Determine the remote addresses of a remote host
socket  -- Create a socket
connect  -- Connect to the remote host using the socket and address information 

getaddrinfo调用成功,创建addrinfo结构的链接列表,并将给定指针设置为指向第一个。

套接字调用创建一个传出套接字并返回一个描述符(有时称为“文件描述符”),可以与readwrite等一起使用。在这种意义上,它是open的网络模拟打开文件流 - 除了我们尚未将套接字连接到任何东西!

最后,connect 调用尝试连接到远程计算机。我们传递原始套接字描述符以及存储在 addrinfo 结构中的套接字地址信息。存在不同种类的套接字地址结构(例如,IPv4 与 IPv6),其可能需要更多存储器。因此,除了传递指针外,还传递了结构的大小:

// Pull out the socket address info from the addrinfo struct:
connect(sockfd, p->ai_addr, p->ai_addrlen)

如何释放为 addrinfo 结构的链表分配的内存?

作为最顶层addrinfo结构上的清理代码调用freeaddrinfo的一部分:

void freeaddrinfo(struct addrinfo *ai);

如果 getaddrinfo 失败,我可以用strerror打印出错误吗?

没有。getaddrinfo的错误处理有点不同:

  • 返回值 _ 是 _ 的错误代码(即不使用errno
  • 使用gai_strerror获取等效的短英文错误文本:
int result = getaddrinfo(...);
if(result) { 
   const char *mesg = gai_strerror(result); 
   ...
}

我可以只请求 IPv4 或 IPv6 连接吗?仅 TCP?

是!使用传递给getaddrinfo的 addrinfo 结构来定义您想要的连接类型。

例如,要通过 IPv6 指定基于流的协议:

struct addrinfo hints;
memset(hints, 0, sizeof(hints));

hints.ai_family = AF_INET6; // Only want IPv6 (use AF_INET for IPv4)
hints.ai_socktype = SOCK_STREAM; // Only want stream-based connection

那些使用gethostbyname的代码示例呢?

旧函数gethostbyname已弃用;这是将主机名转换为 IP 地址的旧方法。端口地址仍需要使用 htons 功能手动设置。使用较新的getaddrinfo编写支持 IPv4 和 IPv6 的代码要容易得多

这很容易!?

是的,不是。创建一个简单的 TCP 客户端很容易 - 但网络通信提供了许多不同的抽象级别,并且可以在每个抽象级别设置几个属性和选项(例如我们没有谈到可以操作选项的setsockopt插座)。有关详细信息,请参阅指南