Java中InetAddress的使用(一):域名解析【享学Java】(上)

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: Java中InetAddress的使用(一):域名解析【享学Java】(上)

前言


Java具有较好的网络编程模型/库,其中非常重要的一个API便是InetAddress。在在java.net网络编程中中有许多类都使用到了InetAddress,包括ServerSocket,Socket,DatagramSocket等等。


你要进行网络编程就得有IP地址、域名、主机等要素,而一个InetAddress里就保存着IP地址,同时还可能包含主机名,并且它提供了主机名 - IP地址互转的方法(比简单的域名解析还牛有木有),本来主要就介绍它看看能够怎么玩。


关于常用的网络概念,请提前做功课:一文搞懂常用的网络概念:域名、静态IP和动态IP、域名解析DNS、动态域名解析DDNS


正文


域名你不陌生,IP地址你也不陌生,但域名解析或许你有些陌生。Java并不希望使用者了解过多的DNS相关知识,因此使用了InetAddress来完成域名 - IP地址的互转工作。


Java域名解析


Java提供InetAddress类(有Inet4Address和Inet6Address两种实现),可以对域名-IP进行正向、逆向解析。InetAddress解析的时候一般是调用系统自带的DNS程序。


比如:Linux下默认使用哪个DNS去解析以及其规则是由/etc/resolv.conf该文件制定的(文件的内容如上示例内容)


对于有些域名,例如www.baidu.com,在不同地区拥有不同的IP;因此使用不同的DNS服务器进行解析,得到的IP一般也不一样。


简单的说:你所处的位置不同,解析www.baidu.com得到的IP地址也就是不一样的


InetAddress


IP地址是IP使用的32位(IPv4)或者128位(IPv6)位无符号数字,它是传输层协议TCP,UDP的基础。InetAddress是Java对IP地址的封装


java.net.IntAddress类是Java对IP地址的高层表示。大多数其它网络类都要用到这个类,包括Socket、ServerSocket、URL、DatagramSocket、DatagramPacket等。InetAddress的实例对象包含了IP地址,同时还可能包含主机名(如果使用主机名来获取InetAddress的实例,或者使用数字来构造,并且启用了反向主机名解析的功能)。InetAddress类提供了将主机名解析为IP地址(或反之)的方法。


InetAddress对域名进行解析是使用本地机器配置(如域名系统DNS和网络信息服务(Network Information Service,NIS))来实现。本地需要向DNS服务器发送查询的请求,然后服务器根据一系列的操作,返回对应的IP地址,为了提高效率,通常本地会缓存一些主机名与IP地址的映射,这样访问相同的地址,就不需要重复发送DNS请求了。


在java.net.InetAddress类同样采用了这种策略。在默认情况下,会缓存一段有限时间的映射,对于主机名解析不成功的结果,会缓存非常短的时间(10秒)来提高性能和准确性。


静态方法得到InetAddress实例


InetAddress并没有提供public的构造器,而是提供了6个静态方法让你构造实例:


public static InetAddress[] getAllByName(String host);
public static InetAddress getByName(String host);
public static InetAddress getByAddress(String host, byte[] addr);
public static InetAddress getByAddress(byte[] addr);
public static InetAddress getLoopbackAddress();
public static InetAddress getLocalHost();


下面分别进行解释说明。


说明:一些的解析结果你可能和我不一样,因为即使对于同一个域名,在不同地方,设置不同时刻解析出来的IP也有可能是不一样的。


  1. public static InetAddress[] getAllByName(String host):给定主机名,返回其IP地址的数组,基于系统配置的DNS服务解析。当然host可以是主机名(域名)或或者是ip地址,这里以www.baidu.com为例。


@Test
public void fun3() throws UnknownHostException {
    InetAddress[] inets = InetAddress.getAllByName("www.baidu.com");
    for (InetAddress inet : inets) {
      // www.baidu.com/61.135.169.125
      // www.baidu.com/61.135.169.121
        System.out.println(inet);
    }
}


image.png


  1. public static InetAddress getByName(String host):它的原理是上面的方法 -> InetAddress.getAllByName(host)[0]取值第一个就是它


@Test
public void fun2() throws UnknownHostException {
    // 网络域名
    InetAddress inet = InetAddress.getByName("www.baidu.com");
    System.out.println("域名:" + inet.getHostName()); // 域名:www.baidu.com
    System.out.println("IP地址:" + inet.getHostAddress()); // IP地址:61.135.169.125
    // 本地域名(本机)
    inet = InetAddress.getByName("localhost");
    System.out.println("域名:" + inet.getHostName()); // 域名:localhost
    System.out.println("IP地址:" + inet.getHostAddress()); // IP地址:127.0.0.1
    // 不存在的域名 抛出异常:java.net.UnknownHostException: aaaaaa.com
    // tips:abc.com这种域名是存在的哟
    inet = InetAddress.getByName("aaaaaa.com");
    System.out.println("域名:" + inet.getHostName());
    System.out.println("IP地址:" + inet.getHostAddress());
}


对于此部分的域名解析,有如下注意事项:


  • 对于外网域名的解析(如www.baidu.com),你的机器必须能够访问外网才能解析到IP地址。否则java.net.UnknownHostException
  • 当然若你是在Linux下通过resolv.conf指定了自己的域名解析器,那么到底解析到哪去由你决定(比如你的内网域名都可以被解析了)
  • 域名不能加上协议。若你这么写http://www.baidu.com就抛错UnknownHostException
  • 对于外网域名解析,每个人解析得到的地址可能不一样。比如此处我对www.baidu.com解析得到的地址是61.135.169.125,是因为我在北京所以得到的是北京的一个IP地址


另外,为了方便你在windows里看到DNS缓存的效果,你可以使用这两个命令来查看:


  • ipconfig /displaydns:展示出当前的dns本地缓存
  • ipconfig /flushdns:清空本地缓存


以上两个方法也叫:用域名创建InetAddress对象。这种方式想获得IP的话,必须经过DNS服务解析~


但是请注意:如果你host传入的就是ip地址的话,就不会经过DNS解析了


  1. public static InetAddress getByAddress(String host, byte[] addr):根据提供的主机名以及 IP 地址创建InetAddress
@Test
public void fun0() throws UnknownHostException {
    // 同时指定域名 和 ip地址,那就是自己建立了对应关系喽
    InetAddress inet = InetAddress.getByAddress("www.baidu.com", new byte[]{61, (byte) 135, (byte) 169, 125});
    System.out.println("域名:" + inet.getHostName()); // 域名:www.baidu.com
    System.out.println("IP地址:" + inet.getHostAddress()); // IP地址:61.135.169.125
}
相关文章
|
15天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
45 2
|
19天前
|
Java
轻松上手Java字节码编辑:IDEA插件VisualClassBytes全方位解析
本插件VisualClassBytes可修改class字节码,包括class信息、字段信息、内部类,常量池和方法等。
71 6
|
3天前
|
Java 数据库连接 开发者
Java中的异常处理机制:深入解析与最佳实践####
本文旨在为Java开发者提供一份关于异常处理机制的全面指南,从基础概念到高级技巧,涵盖try-catch结构、自定义异常、异常链分析以及最佳实践策略。不同于传统的摘要概述,本文将以一个实际项目案例为线索,逐步揭示如何高效地管理运行时错误,提升代码的健壮性和可维护性。通过对比常见误区与优化方案,读者将获得编写更加健壮Java应用程序的实用知识。 --- ####
|
6天前
|
数据采集 存储 Web App开发
Java爬虫:深入解析商品详情的利器
在数字化时代,信息处理能力成为企业竞争的关键。本文探讨如何利用Java编写高效、准确的商品详情爬虫,涵盖爬虫技术概述、Java爬虫优势、开发步骤、法律法规遵守及数据处理分析等内容,助力电商领域市场趋势把握与决策支持。
|
11天前
|
存储 安全 Java
Java多线程编程中的并发容器:深入解析与实战应用####
在本文中,我们将探讨Java多线程编程中的一个核心话题——并发容器。不同于传统单一线程环境下的数据结构,并发容器专为多线程场景设计,确保数据访问的线程安全性和高效性。我们将从基础概念出发,逐步深入到`java.util.concurrent`包下的核心并发容器实现,如`ConcurrentHashMap`、`CopyOnWriteArrayList`以及`BlockingQueue`等,通过实例代码演示其使用方法,并分析它们背后的设计原理与适用场景。无论你是Java并发编程的初学者还是希望深化理解的开发者,本文都将为你提供有价值的见解与实践指导。 --- ####
|
12天前
|
Java 测试技术 API
Java 反射机制:深入解析与应用实践
《Java反射机制:深入解析与应用实践》全面解析Java反射API,探讨其内部运作原理、应用场景及最佳实践,帮助开发者掌握利用反射增强程序灵活性与可扩展性的技巧。
|
17天前
|
存储 算法 Java
Java Set深度解析:为何它能成为“无重复”的代名词?
Java的集合框架中,Set接口以其“无重复”特性著称。本文解析了Set的实现原理,包括HashSet和TreeSet的不同数据结构和算法,以及如何通过示例代码实现最佳实践。选择合适的Set实现类和正确实现自定义对象的hashCode()和equals()方法是关键。
25 4
|
20天前
|
Java 编译器 数据库连接
Java中的异常处理机制深度解析####
本文深入探讨了Java编程语言中异常处理机制的核心原理、类型及其最佳实践,旨在帮助开发者更好地理解和应用这一关键特性。通过实例分析,揭示了try-catch-finally结构的重要性,以及如何利用自定义异常提升代码的健壮性和可读性。文章还讨论了异常处理在大型项目中的最佳实践,为提高软件质量提供指导。 ####
|
24天前
|
存储 设计模式 分布式计算
Java中的多线程编程:并发与并行的深度解析####
在当今软件开发领域,多线程编程已成为提升应用性能、响应速度及资源利用率的关键手段之一。本文将深入探讨Java平台上的多线程机制,从基础概念到高级应用,全面解析并发与并行编程的核心理念、实现方式及其在实际项目中的应用策略。不同于常规摘要的简洁概述,本文旨在通过详尽的技术剖析,为读者构建一个系统化的多线程知识框架,辅以生动实例,让抽象概念具体化,复杂问题简单化。 ####
|
23天前
|
存储 分布式计算 Java
存算分离与计算向数据移动:深度解析与Java实现
【11月更文挑战第10天】随着大数据时代的到来,数据量的激增给传统的数据处理架构带来了巨大的挑战。传统的“存算一体”架构,即计算资源与存储资源紧密耦合,在处理海量数据时逐渐显露出其局限性。为了应对这些挑战,存算分离(Disaggregated Storage and Compute Architecture)和计算向数据移动(Compute Moves to Data)两种架构应运而生,成为大数据处理领域的热门技术。
40 2

相关产品

  • 云解析DNS
  • 推荐镜像

    更多