我们总是希望我们的开发/测试环境能够像生产一样,但是我们常常会陷入使用JDK函数的陷阱,这些函数没有非常明确地暴露环境依赖(比如必需的方法参数),或者可能很难检测到意识到。例如:
public String(byte bytes[])
使用默认编码new Date()/Instant.now()
使用sytem timezone System.out.printf("%n")
使用平台线结尾其中一些可以由-Dfile.encoding=UTF-8
等JVM参数驱动。
但是如何找到所有这些问题?
答案 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的一部分。
作为一般建议:阅读您正在使用的API的文档,并了解支持它们的操作系统功能