如果手机时区已更改,SimpleDateFormat setTimeZone无效

时间:2017-12-23 00:02:55

标签: java android timezone jodatime

我在使用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-8)
  • 我解析了指定时区(LA)的日期
  • 然后我使用原始时区(LA)和当前手机时区
  • 格式化日期

我将手机时区设置为芝加哥(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()清楚地反映了此更改。

有没有办法解决这个问题?

1 个答案:

答案 0 :(得分:2)

找到解决方案!

我一开始就致电DateTimeZone.setDefault(DateTimeZone.forID(TimeZone.getDefault().getID()))

我猜应用程序启动时会调用DateTimeZone.setDefault但后来joda-time本身从未重置,即使用户的时区可能已更改。