InetAddress.getLocalHost 执行很慢?( 二 )

InetAddress.getLocalHost 执行很慢?

文章插图
我们再次把hosts中的主机名去掉,并使用 Arthas 工具的 trace 命令看看链路耗时:
InetAddress.getLocalHost 执行很慢?

文章插图
提示:如果抓包时出现 No class or method is affected 的报错,可查看对应的日志文件进行排查,见下图:
InetAddress.getLocalHost 执行很慢?

文章插图
InetAddress.getLocalHost 执行很慢?

文章插图
可知需要提升下权限,执行命令 options unsafe true 后,再尝试使用 trace命令即可 。
但好巧不巧,居然抓不到调用链?那我们试试用 Arthas 的 profiler 命令生成一下火焰图吧:
InetAddress.getLocalHost 执行很慢?

文章插图
可以看到很多编译相关的,我们忽略之,只把主机信息获取的那部分放大:
InetAddress.getLocalHost 执行很慢?

文章插图
哦吼,时间基本都耗在了 InetAddress.getAddressesFromNameService 这行代码:
InetAddress.getLocalHost 执行很慢?

文章插图
往下追溯,可知时间基本耗在了 nameService.lookupAllHostAddr:
InetAddress.getLocalHost 执行很慢?

文章插图
InetAddress.getLocalHost 执行很慢?

文章插图
再往下就到了native方法:
InetAddress.getLocalHost 执行很慢?

文章插图
于是我们到 jdk 源码中看看(我用的 jdk8):
InetAddress.getLocalHost 执行很慢?

文章插图
接下来需要找 getaddrinfo 的实现,由于不知道具体的实现源码在哪里,于是我们在网上找一下 Linux 系统的源码作为参考,参见:https://codebrowser.dev/glibc/glibc/sysdeps/posix/getaddrinfo.c.html#getaddrinfo
InetAddress.getLocalHost 执行很慢?

文章插图
内部的具体实现基本都是和操作系统交互,我们简单瞄几眼就行 。另外,在 getaddrinfo 源码中没有找到火焰图给出的调用链,我们暂时不再深入 。
InetAddress.getLocalHost 执行很慢?

文章插图
目前,我们知道了方法 getaddrinfo 会被调用,因此简单写段 c 程序复现一下:
#include<sys/time.h>#include <iostream>#include <string.h>#include <sys/types.h>#include <sys/socket.h>#include <netdb.h>#include <netinet/in.h>#include <arpa/inet.h>using namespace std;int main(){char* hostname = "xiaoxi666s-MacBook-Pro.local";addrinfo hints, *res;in_addr addr;int err;struct timeval start, end;gettimeofday(&start, NULL);memset(&hints, 0, sizeof(addrinfo));hints.ai_socktype = SOCK_STREAM;hints.ai_family = AF_INET;if((err = getaddrinfo(hostname, NULL, &hints, &res)) != 0){// 打印耗时(异常情况)gettimeofday(&end, NULL);printf("times=%d\n", end.tv_usec - start.tv_usec);printf("error %d : %s\n", err, gai_strerror(err));return 1;}// 打印耗时(正常情况)gettimeofday(&end, NULL);printf("times=%d\n", end.tv_usec - start.tv_usec);addr.s_addr = ((sockaddr_in*)(res->ai_addr))->sin_addr.s_addr;printf("ip addresss: %s\n", inet_ntoa(addr));freeaddrinfo(res);return 0;}
其中的 hostname 即为主机名 xiaoxi666s-MacBook-Pro.local,我们在 Java 项目中调试时也可以看到,上面的程序中直接将其写死 。
运行程序,对比下 hosts 文件中 没有添加主机名 和 添加主机名后的输出结果:
# hosts 文件中没有添加主机名times=6431error 8 : nodename nor servname provided, or not known

经验总结扩展阅读