我在使用SimpleDateFormat和时区时遇到了一个奇怪的问题。
基本上,我有这段代码:
String input = "2017-12-21 16:15:00";
String inputTZ = "America/Los_Angeles";
String phoneTZ = TimeZone.getDefault().getID();
SimpleDateFormat fmtInput = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss", Locale.US);
fmtInput.setTimeZone(TimeZone.getTimeZone(inputTZ));
SimpleDateFormat fmtOutputEventTZ = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss", Locale.US);
fmtOutputEventTZ.setTimeZone(TimeZone.getTimeZone(inputTZ));
SimpleDateFormat fmtOutputPhoneTZ = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss", Locale.US);
fmtOutputPhoneTZ.setTimeZone(TimeZone.getTimeZone(phoneTZ));
try {
LocalDateTime dt = new LocalDateTime(fmtInput.parse(input));
Log.d("DEBUG>>>>", "INPUT=" + input);
Log.d("DEBUG>>>>", "TZ_EVENT=" + inputTZ);
Log.d("DEBUG>>>>", "TZ_PHONE=" + phoneTZ);
Log.d("DEBUG>>>>", "DT=" + dt);
Log.d("DEBUG>>>>", "OUTPUT_EVENT=" + fmtOutputEventTZ.format(dt.toDate()));
Log.d("DEBUG>>>>", "OUTPUT_PHONE=" + fmtOutputPhoneTZ.format(dt.toDate()));
}
catch (Exception e) {
e.printStackTrace();
}
代码逻辑如下:
我将手机时区设置为芝加哥(GMT-6)运行代码并获得以下输出。
D/DEBUG>>>>: INPUT=2017-12-21 16:15:00
D/DEBUG>>>>: TZ_EVENT=America/Los_Angeles
D/DEBUG>>>>: TZ_PHONE=America/Chicago
D/DEBUG>>>>: DT=2017-12-21T18:15:00.000
D/DEBUG>>>>: OUTPUT_EVENT=2017-12-21 16:15:00
D/DEBUG>>>>: OUTPUT_PHONE=2017-12-21 18:15:00
基本上,结果是预期的结果。
但是,如果我在不重新启动应用的情况下将手机时区更改为纽约(GMT-5),我会得到以下输出
D/DEBUG>>>>: INPUT=2017-12-21 16:15:00
D/DEBUG>>>>: TZ_EVENT=America/Los_Angeles
D/DEBUG>>>>: TZ_PHONE=America/New_York
D/DEBUG>>>>: DT=2017-12-21T18:15:00.000
D/DEBUG>>>>: OUTPUT_EVENT=2017-12-21 15:15:00
D/DEBUG>>>>: OUTPUT_PHONE=2017-12-21 18:15:00
请注意TimeZone.getDefault().getID()
返回的时区是正确的,但解析时,SimpleDateFormat将其转换为电话时区仍为芝加哥,从而返回错误的时间!
但是,如果我杀了应用程序并重新启动它,相同的代码工作正常:
D/DEBUG>>>>: INPUT=2017-12-21 16:15:00
D/DEBUG>>>>: TZ_EVENT=America/Los_Angeles
D/DEBUG>>>>: TZ_PHONE=America/New_York
D/DEBUG>>>>: DT=2017-12-21T19:15:00.000
D/DEBUG>>>>: OUTPUT_EVENT=2017-12-21 16:15:00
D/DEBUG>>>>: OUTPUT_PHONE=2017-12-21 19:15:00
根据我的理解,SimpleDateFormat.parse
解析日期并将其存储为自1970年以来的秒数。使用setTimeZone
应该使用时区来适当地调整时间。
然而,似乎电话时区更改未被SimpleDateFormat反映,而TimeZone.getDefault().getID()
清楚地反映了此更改。
有没有办法解决这个问题?
答案 0 :(得分:2)
找到解决方案!
我一开始就致电DateTimeZone.setDefault(DateTimeZone.forID(TimeZone.getDefault().getID()))
。
我猜应用程序启动时会调用DateTimeZone.setDefault
但后来joda-time本身从未重置,即使用户的时区可能已更改。