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

时间:2016-08-28 08:02:36

标签: java jvm cross-platform platform

我们总是希望我们的开发/测试环境能够像生产一样,但是我们常常会陷入使用JDK函数的陷阱,这些函数没有非常明确地暴露环境依赖(比如必需的方法参数),或者可能很难检测到意识到。例如:

  • public String(byte bytes[])使用默认编码
  • new Date()/Instant.now()使用sytem timezone
  • System.out.printf("%n")使用平台线结尾

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

但是如何找到所有这些问题?

3 个答案:

答案 0 :(得分:2)

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

对于几乎完整的列表,您可以运行此检查,其中列出了操作系统环境变量和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);

答案 1 :(得分:2)

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

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

答案 2 :(得分:2)

因为您要求提供完整列表:每次与操作系统或本机库交互时。

每当你不做纯计算时,忽略严格的数学就会有差异。有时它们是次要的,只发生在边缘情况下,有时它们明确地成为API的一部分。

  • 开始一个新主题?受OS限制。
  • 想要以纳秒精度安排任务吗?取决于平台上可用的计时器精度。
  • 文件名?取决于文件系统支持的内容。
  • 打开插座?可用端口范围由OS配置。
  • ...

作为一般建议:阅读您正在使用的API的文档,并了解支持它们的操作系统功能