下面哪一种方法是在Java中获得当前计算机主机名的最佳和最可移植的方法?

Runtime.getRuntime().exec(“hostname”)

vs

InetAddress.getLocalHost().getHostName()


当前回答

虽然这个话题已经有了答案,但还有更多要说的。

首先,显然我们需要一些定义。InetAddress.getLocalHost(). gethostname()提供了从网络角度看的主机名称。这种方法的问题在其他答案中有很好的记录:它通常需要DNS查找,如果主机有多个网络接口,它是不明确的,有时它只是失败(见下文)。

但在任何操作系统上都有另一个名称。在引导过程的早期定义的主机名,远早于网络初始化。Windows将其称为计算机名,Linux将其称为内核主机名,Solaris则使用节点名。我最喜欢计算机名这个词,所以从现在开始我就用这个词了。

查找计算机名

在Linux/Unix上,计算机名是从C函数gethostname()中获得的,或者是从shell中获得的主机名命令或Bash-like shell中的hostname环境变量。 在Windows上,计算机名是从环境变量computername或Win32 GetComputerName函数中获得的。

Java没有办法获得我定义的“计算机名”。当然,在其他答案中也有一些解决办法,比如Windows调用System.getenv("COMPUTERNAME"),但在Unix/Linux上,如果不求助于JNI/JNA或Runtime.exec(),就没有好的解决办法。如果您不介意使用JNI/JNA解决方案,那么可以使用gethostname4j,它非常简单,非常易于使用。

让我们继续看两个例子,一个来自Linux,一个来自Solaris,它们演示了如何很容易地陷入无法使用标准Java方法获取计算机名的情况。

Linux示例

在新创建的系统上,安装期间的主机被命名为“chicago”,现在我们更改所谓的内核主机名:

$ hostnamectl --static set-hostname dallas

现在内核主机名是'dallas',从hostname命令可以看出:

$ hostname
dallas

但是我们还有

$ cat /etc/hosts
127.0.0.1   localhost
127.0.0.1   chicago

这里没有配置错误。它只是意味着主机的网络名称(或者更确切地说,环回接口的名称)与主机的计算机名称不同。

现在,尝试执行InetAddress.getLocalHost(). gethostname(),它将抛出java.net.UnknownHostException。你基本上被困住了。没有办法检索值'dallas'或值'chicago'。

Solaris的例子

下面的示例基于Solaris 11.3。

主机已经过特意配置,使环回名称<>节点名。

换句话说我们有:

$ svccfg -s system/identity:node listprop config
...
...
config/loopback             astring        chicago
config/nodename             astring        dallas

/etc/hosts的内容:

:1 chicago localhost
127.0.0.1 chicago localhost loghost

hostname命令的结果是:

$ hostname
dallas

就像在Linux示例中调用InetAddress.getLocalHost().getHostName()将失败

java.net.UnknownHostException: dallas:  dallas: node name or service name not known

就像Linux示例一样,您现在陷入了困境。没有办法检索值'dallas'或值'chicago'。

什么时候你才会真正为此而挣扎?

通常你会发现InetAddress.getLocalHost(). gethostname()确实会返回一个等于计算机名的值。因此没有问题(除了名称解析的额外开销)。

这个问题通常出现在PaaS环境中,其中计算机名和环回接口名之间存在差异。例如,人们报告亚马逊EC2中的问题。

错误/ RFE报告

稍微搜索一下就可以看到这个RFE报告:link1, link2。然而,从对该报告的评论来看,这个问题似乎在很大程度上被JDK团队误解了,所以它不太可能得到解决。

我喜欢RFE与其他编程语言的比较。

其他回答

环境变量也可以提供有用的方法——Windows上的COMPUTERNAME,大多数现代Unix/Linux shell上的HOSTNAME。

参见:https://stackoverflow.com/a/17956000/768795

我使用这些作为InetAddress.getLocalHost(). gethostname()的“补充”方法,因为正如一些人指出的那样,该函数不能在所有环境中工作。

Runtime.getRuntime().exec("hostname")是另一个可能的补充。在这个阶段,我还没有用过它。

import java.net.InetAddress;
import java.net.UnknownHostException;

// try InetAddress.LocalHost first;
//      NOTE -- InetAddress.getLocalHost().getHostName() will not work in certain environments.
try {
    String result = InetAddress.getLocalHost().getHostName();
    if (StringUtils.isNotEmpty( result))
        return result;
} catch (UnknownHostException e) {
    // failed;  try alternate means.
}

// try environment properties.
//      
String host = System.getenv("COMPUTERNAME");
if (host != null)
    return host;
host = System.getenv("HOSTNAME");
if (host != null)
    return host;

// undetermined.
return null;

在Java中获取当前计算机主机名的最方便的方法如下:

import java.net.InetAddress;
import java.net.UnknownHostException;

public class getHostName {

    public static void main(String[] args) throws UnknownHostException {
        InetAddress iAddress = InetAddress.getLocalHost();
        String hostName = iAddress.getHostName();
        //To get  the Canonical host name
        String canonicalHostName = iAddress.getCanonicalHostName();

        System.out.println("HostName:" + hostName);
        System.out.println("Canonical Host Name:" + canonicalHostName);
    }
}
hostName == null;
Enumeration<NetworkInterface> interfaces = NetworkInterface.getNetworkInterfaces();
{
    while (interfaces.hasMoreElements()) {
        NetworkInterface nic = interfaces.nextElement();
        Enumeration<InetAddress> addresses = nic.getInetAddresses();
        while (hostName == null && addresses.hasMoreElements()) {
            InetAddress address = addresses.nextElement();
            if (!address.isLoopbackAddress()) {
                hostName = address.getHostName();
            }
        }
    }
}

虽然这个话题已经有了答案,但还有更多要说的。

首先,显然我们需要一些定义。InetAddress.getLocalHost(). gethostname()提供了从网络角度看的主机名称。这种方法的问题在其他答案中有很好的记录:它通常需要DNS查找,如果主机有多个网络接口,它是不明确的,有时它只是失败(见下文)。

但在任何操作系统上都有另一个名称。在引导过程的早期定义的主机名,远早于网络初始化。Windows将其称为计算机名,Linux将其称为内核主机名,Solaris则使用节点名。我最喜欢计算机名这个词,所以从现在开始我就用这个词了。

查找计算机名

在Linux/Unix上,计算机名是从C函数gethostname()中获得的,或者是从shell中获得的主机名命令或Bash-like shell中的hostname环境变量。 在Windows上,计算机名是从环境变量computername或Win32 GetComputerName函数中获得的。

Java没有办法获得我定义的“计算机名”。当然,在其他答案中也有一些解决办法,比如Windows调用System.getenv("COMPUTERNAME"),但在Unix/Linux上,如果不求助于JNI/JNA或Runtime.exec(),就没有好的解决办法。如果您不介意使用JNI/JNA解决方案,那么可以使用gethostname4j,它非常简单,非常易于使用。

让我们继续看两个例子,一个来自Linux,一个来自Solaris,它们演示了如何很容易地陷入无法使用标准Java方法获取计算机名的情况。

Linux示例

在新创建的系统上,安装期间的主机被命名为“chicago”,现在我们更改所谓的内核主机名:

$ hostnamectl --static set-hostname dallas

现在内核主机名是'dallas',从hostname命令可以看出:

$ hostname
dallas

但是我们还有

$ cat /etc/hosts
127.0.0.1   localhost
127.0.0.1   chicago

这里没有配置错误。它只是意味着主机的网络名称(或者更确切地说,环回接口的名称)与主机的计算机名称不同。

现在,尝试执行InetAddress.getLocalHost(). gethostname(),它将抛出java.net.UnknownHostException。你基本上被困住了。没有办法检索值'dallas'或值'chicago'。

Solaris的例子

下面的示例基于Solaris 11.3。

主机已经过特意配置,使环回名称<>节点名。

换句话说我们有:

$ svccfg -s system/identity:node listprop config
...
...
config/loopback             astring        chicago
config/nodename             astring        dallas

/etc/hosts的内容:

:1 chicago localhost
127.0.0.1 chicago localhost loghost

hostname命令的结果是:

$ hostname
dallas

就像在Linux示例中调用InetAddress.getLocalHost().getHostName()将失败

java.net.UnknownHostException: dallas:  dallas: node name or service name not known

就像Linux示例一样,您现在陷入了困境。没有办法检索值'dallas'或值'chicago'。

什么时候你才会真正为此而挣扎?

通常你会发现InetAddress.getLocalHost(). gethostname()确实会返回一个等于计算机名的值。因此没有问题(除了名称解析的额外开销)。

这个问题通常出现在PaaS环境中,其中计算机名和环回接口名之间存在差异。例如,人们报告亚马逊EC2中的问题。

错误/ RFE报告

稍微搜索一下就可以看到这个RFE报告:link1, link2。然而,从对该报告的评论来看,这个问题似乎在很大程度上被JDK团队误解了,所以它不太可能得到解决。

我喜欢RFE与其他编程语言的比较。

只有一句话……跨平台(Windows-Linux-Unix-Mac(Unix))[始终工作,不需要DNS]:

String hostname = new BufferedReader(
    new InputStreamReader(Runtime.getRuntime().exec("hostname").getInputStream()))
   .readLine();

你完蛋了!!