我如何在JDK中找到与平台相关的陷阱的完整列表,如时区,编码,行结尾等



我们总是希望我们的开发/测试环境与生产环境相似,但我们经常会陷入使用JDK函数的陷阱,这些函数没有非常明确地暴露环境依赖关系(如所需的方法参数),或者可能难以检测和意识到。例如:

  • public String(byte bytes[])使用默认编码
  • new Date()/Instant.now()使用系统时区
  • System.out.printf("%n")使用平台行结尾

其中一些可以由JVM参数驱动,如-Dfile.encoding=UTF-8

但是如何找到所有这些陷阱呢?

陷阱的类型取决于应用程序的类型。

对于一个几乎完整的列表,您可以运行这个检查,它列出了操作系统环境变量和java系统属性:

import java.awt.GraphicsEnvironment;
import java.util.Map;
import java.util.Properties;
public class Sof39189179 {
    public static void main(String[] args) {
        Map<String, String> sysenv = System.getenv();
        for(String key: sysenv.keySet())
            System.out.println( key +  ": " + sysenv.get(key));
        Properties properties = System.getProperties();
        for(Object key: properties.keySet())
            System.out.println(key + ": " + properties.get(key));
        GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment(); 
        System.out.println("headless: " + ge.isHeadless());
    }
}
在我的计算机上,我得到以下输出(我省略了一些行):
M2: $M2_HOME/bin
JAVA_HOME: /opt/local/jdk-1.8.0_51
LANG: en_US.UTF-8
CATALINA_HOME: /opt/local/tomcat-8.0.24
...
---------------------
java.runtime.name: Java(TM) SE Runtime Environment
sun.boot.library.path: /opt/local/jdk-1.8.0_51/jre/lib/amd64
java.vm.version: 25.51-b03
java.vm.vendor: Oracle Corporation
java.vendor.url: http://java.oracle.com/
path.separator: :
java.vm.name: Java HotSpot(TM) 64-Bit Server VM
file.encoding.pkg: sun.io
user.country: US
sun.java.launcher: SUN_STANDARD
sun.os.patch.level: unknown
java.vm.specification.name: Java Virtual Machine Specification
user.dir: /home/rudolf/workspace/neon/sof39189179
java.runtime.version: 1.8.0_51-b16
java.awt.graphicsenv: sun.awt.X11GraphicsEnvironment
java.endorsed.dirs: /opt/local/jdk-1.8.0_51/jre/lib/endorsed
os.arch: amd64
java.io.tmpdir: /tmp
line.separator: 
java.vm.specification.vendor: Oracle Corporation
os.name: Linux
...
java.awt.printerjob: sun.print.PSPrinterJob
file.encoding: UTF-8
java.specification.version: 1.8
java.class.path: /home/rudolf/workspace/neon/sof39189179/bin
user.name: rudolf
java.vm.specification.version: 1.8
java.home: /opt/local/jdk-1.8.0_51/jre
sun.arch.data.model: 64
user.language: en
java.specification.vendor: Oracle Corporation
awt.toolkit: sun.awt.X11.XToolkit
java.vm.info: mixed mode
java.version: 1.8.0_51
java.ext.dirs: /opt/local/jdk-1.8.0_51/jre/lib/ext:/usr/java/packages/lib/ext
sun.boot.class.path: /opt/local/jdk-1.8.0_51/jre/lib/resources.jar:/opt/local/jdk-1.8.0_51/jre/lib/rt.jar:/opt/local/jdk-1.8.0_51/jre/lib/sunrsasign.jar:/opt/local/jdk-1.8.0_51/jre/lib/jsse.jar:/opt/local/jdk-1.8.0_51/jre/lib/jce.jar:/opt/local/jdk-1.8.0_51/jre/lib/charsets.jar:/opt/local/jdk-1.8.0_51/jre/lib/jfr.jar:/opt/local/jdk-1.8.0_51/jre/classes
java.vendor: Oracle Corporation
file.separator: /
java.vendor.url.bug: http://bugreport.sun.com/bugreport/
sun.io.unicode.encoding: UnicodeLittle
sun.cpu.endian: little
sun.cpu.isalist: 
headless: false

您可以从这个列表中确定哪些属性与您的情况相关,并检查您的代码。或者,更好的是,在代码中考虑到这些属性。

System.out.printf("%n")使用平台行结尾

我更喜欢使用上面列出的属性,例如,对于打印换行符,我使用如下:

String newline = (String) System.getProperties().get("line.separator");
System.out.println("newline: " +  newline);

由于您要求的是完整列表:每次与操作系统或本机库交互时。

每次你不做纯计算让我们忽略严格的数学就会有区别。有时它们是次要的,只出现在边缘情况下,有时它们是API的显式组成部分。

    启动一个新线程?受操作系统限制。
  • 想要以纳秒级的精度调度任务?根据平台上可用的计时器精度。
  • 文件名吗?这取决于文件系统支持什么。
  • 打开套接字?可用端口范围由操作系统配置。

一般建议:阅读你正在使用的api的文档,了解支持它们的操作系统特性

不是你问题的直接答案,但-XX:+PrintFlagsFinal-XshowSettings:all在比较不同的环境方面有很大帮助。这些标志将显示默认的(取决于硬件/操作系统)jvm设置和env/system属性。

您还需要比较操作系统设置,如环境变量,操作系统限制/配额,安全设置,文件系统设置等。

最新更新