如何在Joda时间将日期时间从时区A转换为时区B?

时间:2015-03-15 12:53:25

标签: java jodatime

想象一下,我有一个来自一个系统的日期时间值,以及来自另一个系统的时区信息(源时区和目标时区)。我需要将日期时间从源时间转换为目标时区。

例如,我们说我在莫斯科时区获得了一个约会时间。我需要把它转换成柏林时区。我怎么能用Joda Time做到这一点?

private static final String TIMEZONE_BERLIN_ID = "Europe/Berlin"; //$NON-NLS-1$

private static final DateTimeZone TIMEZONE_BERLIN = DateTimeZone
    .forID(TIMEZONE_BERLIN_ID);

private static final DateTimeZone TIMEZONE_MOSCOW = DateTimeZone
    .forID("Europe/Moscow"); //$NON-NLS-1$

@Test
public void testDateTimeConversion() {
    final Date sourceDateTime = new DateTime(2015, 3, 15, 12, 55).toDate();
    // We assume that sourceDateTime is in TIMEZONE_MOSCOW. We want to convert it to TIMEZONE_BERLIN.
    // Note that the conversion should work regardless whether sourceDateTime contains any timezone
    // information. It must be converted from TIMEZONE_MOSCOW to TIMEZONE_BERLIN (regardless of whether
    // a timezone is specified in it).
    final Date expectedResult =
        new DateTime(2015, 3, 15, 12 - 2, 55).toDate(); // At 12:55 in Moscow, it is 10:55 in Berlin.
    final Date actualResult =
        new DateTime(sourceDateTime, TIMEZONE_MOSCOW).toDateTime(
            TIMEZONE_BERLIN).toDate();
    Assert.assertEquals(expectedResult, actualResult);
}

为什么actualResult包含错误的日期时间?

new DateTime(sourceDateTime, TIMEZONE_MOSCOW).withZone(TIMEZONE_BERLIN).toDate()也不起作用。

3 个答案:

答案 0 :(得分:2)

java.util.Date对象不包含时区信息。从格林威治标准时间01-01-1970 00:00:00开始,它只是一个几毫秒的包装器。您不能拥有特定时区中的Date对象。 Date对象总是指"绝对"时间点。您不应假装Date对象包含特定时区(例如莫斯科或柏林时区)的日期和时间。

相反,您可以在格式化Date对象进行显示时指定时区,方法是设置要在Date对象上查看DateFormat的时区:

DateFormat df1 = new SimpleDateFormat("dd-MM-yyyy HH:mm:ss z");
df1.setTimeZone(TimeZone.getTimeZone("Europe/Berlin"));

DateFormat df2 = new SimpleDateFormat("dd-MM-yyyy HH:mm:ss z");
df2.setTimeZone(TimeZone.getTimeZone("Europe/Moscow"));

Date now = new Date();

// Note: The same Date object, but displayed in different timezones
System.out.println("The time in Berlin: " + df1.format(now));
System.out.println("The time in Moscow: " + df2.format(now));

答案 1 :(得分:1)

这个似乎有效:

@Test
public void testDateTimeConversion() {
    final Date sourceDateTime = new DateTime(2015, 3, 15, 12, 55, TIMEZONE_MOSCOW).toDate();
    final Date expectedResult =
        new DateTime(2015, 3, 15, 12 - 2, 55, TIMEZONE_BERLIN).toLocalDateTime().toDate();

    final DateTime timeInMoscow =
        new DateTime(sourceDateTime, TIMEZONE_MOSCOW);
    final DateTime timeInBerlin = timeInMoscow.toDateTime(TIMEZONE_BERLIN);
    final Date actualResult = timeInBerlin.toLocalDateTime().toDate();

    Assert.assertEquals(expectedResult, actualResult);
}

答案 2 :(得分:0)

我认为这个可以解决您的问题。只需将此逻辑转换为JUnit然后它就可以工作。

    DateTimeZone TIMEZONE_BERLIN = DateTimeZone.forID("Europe/Berlin");
    DateTimeZone TIMEZONE_MOSCOW = DateTimeZone.forID("Europe/Moscow");

    //moscow timezone 12:55
    DateTime sourceTimeMoscow = new DateTime(2015, 3, 15, 12, 55, TIMEZONE_MOSCOW);
    //berlin time expected 10:55
    Date expectedResult = new DateTime(2015, 3, 15, 10, 55, TIMEZONE_BERLIN).toLocalDateTime().toDate();
    //get actual result in Berlin in reference to Moscow
    Date actualResult = new DateTime(sourceTimeMoscow, TIMEZONE_BERLIN).toLocalDateTime().toDate();

    //for display purposes
    System.out.println(expectedResult); //expected result Berlin Time
    System.out.println(actualResult); //actual result Berlin Time

我希望此解决方案可以解决您的问题。