摘要:要探索JDK的核心底层源码,那必须掌握native用法。文章中会以“获取系统的默认时区”为例,介绍说明如何查看native对应方法的源码。

本文分享自华为云社区《要探索JDK的核心底层源码,那必须掌握native用法》,作者: 小虚竹 。

场景

有探索欲的同学,应该会跟我一样,在看JDK源码时,跟到最后,会出现native方法,类似下面这个方法

 /*** Gets the platform defined TimeZone ID.**/private static native String getSystemTimeZoneID(String javaHome);

看到这个native ,说明已经挖到核心了,到了这一步,还是不清楚是怎么获取系统的默认时区的,那怎么办,JDK代码只能跟到这里。

转战OpenJDK,源码下载方式:openjdk: OpenJDK做为GPL许可(GPL-licensed)的Java平台的实现,Sun正式发布它已经一年有余

什么是native

native是一个计算机函数,一个Native Method就是一个Java调用非Java代码的接口。方法的实现由非Java语言实现,比如C或C++。

native的源码怎么看呢

以**private static native String getSystemTimeZoneID(String javaHome)**为例

getSystemTimeZoneID方法所在的package java.util.TimeZone;

如图所示,找到TimeZone.c下的getSystemTimeZoneID方法

/** Gets the platform defined TimeZone ID*/
JNIEXPORT jstring JNICALL
Java_java_util_TimeZone_getSystemTimeZoneID(JNIEnv *env, jclass ign,jstring java_home, jstring country)
{const char *cname;const char *java_home_dir;char *javaTZ;if (java_home == NULL)return NULL;java_home_dir = JNU_GetStringPlatformChars(env, java_home, 0);if (java_home_dir == NULL)return NULL;if (country != NULL) {cname = JNU_GetStringPlatformChars(env, country, 0);/* ignore error cases for cname */} else {cname = NULL;}/** Invoke platform dependent mapping function*/javaTZ = findJavaTZ_md(java_home_dir, cname);free((void *)java_home_dir);if (cname != NULL) {free((void *)cname);}if (javaTZ != NULL) {jstring jstrJavaTZ = JNU_NewStringPlatform(env, javaTZ);free((void *)javaTZ);return jstrJavaTZ;}return NULL;
}

重点:调用不同平台相关的映射函数

  /** Invoke platform dependent mapping function*/javaTZ = findJavaTZ_md(java_home_dir, cname);

去查找findJavaTZ_md方法时,发现存在分别在solaris和windows两个目录下。

查了下这两个目录的差别:

因为OpenJDK里,Java标准库和部分工具的源码repo(jdk目录)里,BSD和Linux的平台相关源码都是在solaris目录里的。
原本Sun JDK的源码里平台相关的目录就是从solaris和windows这两个目录开始的,后来Unix系的平台相关代码全都放在solaris目录下了,共用大部分代码。作者:RednaxelaFX
链接:https://www.zhihu.com/question/58982441/answer/170264788
来源:知乎

简单的理解就是:

  • window系统下,使用windows目录下编译的JDK代码
  • unix系的平台下,使用solaris目录下编译的JDK代码

了解不同系统下findJavaTZ_md方法执行

windows系统

/** Detects the platform time zone which maps to a Java time zone ID.*/
char *findJavaTZ_md(const char *java_home_dir, const char *country)
{char winZoneName[MAX_ZONE_CHAR];char winMapID[MAX_MAPID_LENGTH];char *std_timezone = NULL;int  result;winMapID[0] = 0;result = getWinTimeZone(winZoneName, winMapID);if (result != VALUE_UNKNOWN) {if (result == VALUE_GMTOFFSET) {std_timezone = _strdup(winZoneName);} else {std_timezone = matchJavaTZ(java_home_dir, result,winZoneName, winMapID, country);}}return std_timezone;
}

注释写得很清楚,获取“Time Zones”注册表中的当前时区

/** Gets the current time zone entry in the "Time Zones" registry.*/
static int getWinTimeZone(char *winZoneName, char *winMapID)
{
...
}

时区的设置方式:

那时区上的选择值是从哪取到的,上面有说了,是在注册表中取值

打开注册表 :Regedit–>

计算机\HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Time Zones\

unix系的平台

findJavaTz_md()方法的注释上写得很清楚了:将平台时区ID映射为Java时区ID

/** findJavaTZ_md() maps platform time zone ID to Java time zone ID* using <java_home>/lib/tzmappings. If the TZ value is not found, it* trys some libc implementation dependent mappings. If it still* can't map to a Java time zone ID, it falls back to the GMT+/-hh:mm* form. `country', which can be null, is not used for UNIX platforms.*/
/*ARGSUSED1*/
char *
findJavaTZ_md(const char *java_home_dir, const char *country)
{char *tz;char *javatz = NULL;char *freetz = NULL;tz = getenv("TZ");#ifdef __linux__if (tz == NULL) {
#else
#ifdef __solaris__if (tz == NULL || *tz == '\0') {
#endif
#endiftz = getPlatformTimeZoneID();freetz = tz;}/** Remove any preceding ':'*/if (tz != NULL && *tz == ':') {tz++;}#ifdef __solaris__if (strcmp(tz, "localtime") == 0) {tz = getSolarisDefaultZoneID();freetz = tz;}
#endifif (tz != NULL) {
#ifdef __linux__/** Ignore "posix/" prefix.*/if (strncmp(tz, "posix/", 6) == 0) {tz += 6;}
#endifjavatz = strdup(tz);if (freetz != NULL) {free((void *) freetz);}}return javatz;
}

步骤:

1、使用< Java home>/lib/tzmappings,。如果没有找到"TZ"变量,就进行第2步

2、 tz = getPlatformTimeZoneID(); 执行Linux特定的映射,如果找到,返回一个时区ID,否则返回null

【Linux】Centos7修改系统时区timezone方式:

timedatectl

修改时区

timedatectl  set-timezone Asia/Shanghai

3、对比/etc/localtime与"/usr/share/zoneinfo目录下的文件,如果一致,就返回时区ID,没有则到第4步

4、返回到GMT

点击关注,第一时间了解华为云新鲜技术~

带你掌握不同平台下,探索JDK源码所需的native方法相关推荐

  1. JDK源码解析——Object的hashCode方法

    目录 前言 说明 一.源码目录结构 (1).JDK目录 (2).hotspot目录 二.基础知识 (1).Object Header(对象头) (2).Lock(锁) 1. 无锁 => 偏向锁 ...

  2. Linux平台下的watchman源码编译

    参考英文:https://facebook.github.io/watchman/docs/install.html 0.准备工作 (1)确定gcc版本 [root@master ~]# gcc -- ...

  3. 如何读jdk源码以及查找linux本地方法,例如DefaultSelectorProvider

    下载openjdk源码 openjdk GitHub源码 https://github.com/openjdk/jdk8 ctrl+N 搜索class,如下得到3个类,选择solaris linux版 ...

  4. Eclipse如何查看jdk源码

    Eclipse 如何查看jdk源码 Eclipse 如何查看jdk源码 点击类名.方法名查看Java的jdk源码 还没有写代码,直接配置jdk源码路径 点击类名.方法名查看Java的jdk源码 当我们 ...

  5. 跨年巨作 13万字 腾讯高工手写JDK源码笔记 带你飙向实战

    灵魂一问,我们为什么要学习JDK源码? 当然不是为了装,毕竟谁没事找事虐自己 ... 1.面试跑不掉.现在只要面试Java相关的岗位,肯定或多或少会会涉及JDK源码相关的问题. 2.弄懂原理才不慌.我 ...

  6. 铁山靠!阿里P9架构师写的这份JDK源码笔记,竟直接带火了GitHub

    众所周知,阅读源码是一件很累的事情,但是同时也是一件能让我们收获很多东西的事情.比如,有些原理搞不懂,但是通过阅读源码就可以让我们恍然大悟. 同时当下的面试也是十分重视考察源码,所以现在去阅读源码已经 ...

  7. 【Java进阶营】膜拜 13万字 腾讯高工手写JDK源码笔记带你飙向实战

    灵魂一问,我们为什么要学习JDK源码? 当然不是为了装,毕竟谁没事找事虐自己 - 1.面试跑不掉.现在只要面试Java相关的岗位,肯定或多或少会会涉及JDK源码相关的问题. 2.弄懂原理才不慌.我们作 ...

  8. 旷世神作,腾讯高工手写13万字JDK源码笔记,从底层远吗 带你飙向实战

    灵魂一问,我们为什么要学习JDK源码? 当然不是为了装,毕竟谁没事找事虐自己 ... 1.面试跑不掉.现在只要面试Java相关的岗位,肯定或多或少会会涉及JDK源码相关的问题. 2.弄懂原理才不慌.我 ...

  9. JDK源码包结构分类

    最近查看JDK源码时,无意间发现几个类在陌生包里:com.sun.*.sun.*.org.*,google了一把总结了下以备他人搜索,如内容有误欢迎指正! Jre库包含的jar文件(jdk1.6):r ...

最新文章

  1. npm install遇到ENOENT: no such file or directory, rename错误
  2. 干货!用大白话告诉你什么是Mock测试
  3. E - 数据结构实验之排序五:归并求逆序数
  4. 插入排序和冒泡排序算法JAVA实现
  5. mysql重新载入my.cnf_怎么重新加载MySQL的my.cnf?
  6. node.js热部署
  7. spring boot中利用mybatis-generator插件生成代码
  8. Java技术:实现pdf和Excel的生成及数据动态插入、导出
  9. java中byte、short、char、boolean实际都是按照int处理的!
  10. 使用Java处理大文件
  11. Selenium3+python 加载Firefox配置
  12. 推荐几个前端 模板 框架,收藏
  13. python用cartopy包画地图_python绘制地图的利器Cartopy使用说明
  14. GPU编程 CUDA C++ 数组归约的示例和解释
  15. 十六进制转ASCII码表
  16. 青龙面板搭配诺兰nvjdc详细教程
  17. 直销系统开发|直销双轨制模式详解
  18. java边缘检测算子代码_图像边缘检测(Canny 算法)的Java实现
  19. 我心依旧之Android Camera模块FW/HAL3探学序
  20. IEEE-CIS Fraud Detection(一)

热门文章

  1. Java 8 API Stream让List操作更便捷
  2. 关于总结记录的一点体会
  3. Python 图片与字符串互转
  4. 万有引力的意思_万有引力和引力有什么不同?四种基本性质力中电磁力最多
  5. splunk 提取字段_全面的Splunk应用日志分析介绍
  6. java getcharat_Java UTF8.getCharArray方法代码示例
  7. Y15BeTa的乱搞方法(占坑待填)
  8. 06旋转数组的最小数字
  9. {Azure} 常用链接
  10. WS-*协议栈及相关概念