想象一下,我有一个来自一个系统的日期时间值,以及来自另一个系统的时区信息(源时区和目标时区)。我需要将日期时间从源时间转换为目标时区。
例如,我们说我在莫斯科时区获得了一个约会时间。我需要把它转换成柏林时区。我怎么能用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()
也不起作用。
答案 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
我希望此解决方案可以解决您的问题。