我的客户端/浏览器在印度,我从javascript获取timezoneoffset 使用以下代码:
var now = new Date();
var localOffSet = now.getTimezoneOffset(); -330 // for India
int localOffSetMin = (localOffSet)*(-1);
我的服务器位于纽约,因此我使用以下方式获取偏移量:
TimeZone timeZone = now.getTimeZone();
int serverOffset = timeZone.getRawOffset();
int serverOffSetMinutes = serverOffset / 60000; // -300 for America/New York
为了在我的机器上找到当地时间,我使用它:
int offSets = Math.abs(serverOffSetMinutes-localOffSetMin);
now.setTime(createDt); // createDt is date field value for some column
now.add(Calendar.MINUTE, offSets); // adds offset
Date localDt = now.getTime();
但我得到的日期/时间比预期时间提前1小时。我错过了什么?
答案 0 :(得分:3)
您可以使用以下代码打印支持的时区列表。
System.out.println(TimeZone.getAvailableIDs().toString());
然后,您可以使用以下代码查找并打印时区之间的差异。您必须注意夏令时。
public void printTimeZoneDifference(String from, String to) {
TimeZone easternStandardTime = TimeZone.getTimeZone(from);
TimeZone indiaStandardTime = TimeZone.getTimeZone(to);
long milliseconds = easternStandardTime.getRawOffset() - indiaStandardTime.getRawOffset() + easternStandardTime.getDSTSavings() - indiaStandardTime.getDSTSavings();
String difference = String.format("%02d min, %02d sec", TimeUnit.MILLISECONDS.toMinutes(milliseconds), TimeUnit.MILLISECONDS.toSeconds(milliseconds) - TimeUnit.MINUTES.toSeconds(TimeUnit.MILLISECONDS.toMinutes(milliseconds)));
System.out.println("The difference in time between" + easternStandardTime.getDisplayName() + " and " + indiaStandardTime.getDisplayName() + " is " + difference);
}
虽然如果我写这样的东西,我可能会将TimeZone对象作为参数传递,并保持该方法单独负责减法。然后我会打印结果或使其成为不同方法的一部分。我没有以这种方式构建帖子,因为我想在帖子中包含所有相关代码。
这种类型的操作已经用Java解决了。如果您正在进行大量的日期操作,那么 Joda时间库可能是您最好的选择。如果你只是在这个实例中操作时间,那么在运行时包含依赖项将会有点过时。
再次打印出TimeZones。
public void printDateTimeZones() {
for(String zone : DateTimeZone.getAvailableIDs()) {
System.out.println(zone);
}
}
然后,您可以使用以下代码的默认格式返回两个DateTimeZones之间的句点(差异)字符串。
public String printPeriod(String from, String to) {
Period period = new Period(new DateTime(DateTimeZone.forID(to)), new DateTime(DateTimeZone.forID(from)));
return PeriodFormat.getDefault().print(period);
}
类似地,Joda提供了一个格式构建器类,允许您指定首选格式。
public String printPeriod(String from, String to) {
PeriodFormatter formatter = new PeriodFormatterBuilder()
.printZeroRarelyFirst()
.appendYears().appendSuffix(" Years").appendSeparator(",")
.appendMonths().appendSuffix(" Months").appendSeparator(",")
.appendWeeks().appendSuffix(" Weeks").appendSeparator(",")
.appendDays().appendSuffix(" Days").appendSeparator(",")
.appendHours().appendSuffix(" Hours").appendSeparator(",")
.appendSeconds().appendSuffix(" Seconds").appendSeparator(",")
.appendMillis().appendSuffix(" Milliseconds")
.toFormatter();
return formatter.print(new Period(new DateTime(DateTimeZone.forID(from)), new DateTime(DateTimeZone.forID(to))));
}
答案 1 :(得分:2)
java.util.Date
个对象has no timezone information。它只有一个long
值,这是1970-01-01T00:00:00Z
的毫秒数(也称为" unix epoch" 或只是&# 34;时期" )。这个值完全独立于时区(你可以说"它在UTC"以及它)。
要将此值转换为其他时区,您不需要在时区之间进行所有这些数学运算。您只需获得此millis值并将其转换为所需的时区。
要从javascript获取值,请执行以下操作:
var d = new Date();
var millis = d.getTime();
变量millis
将包含epoch的毫秒数。在我做过的测试中,此值为1499101493296
。
要创建java.util.Date
对象,只需执行以下操作:
Date date = new Date(1499101493296L);
要在您想要的时区中格式化此日期,请使用SimpleDateFormat
:
SimpleDateFormat sdf = new SimpleDateFormat("dd/MM/yyyy HH:mm:ss");
sdf.setTimeZone(TimeZone.getTimeZone("Asia/Kolkata"));
System.out.println(sdf.format(date));
输出将是:
03/07/2017 22:34:53
如果您需要其他格式,请查看javadoc以获取更多信息。
另请注意,我使用IANA format使用了时区名称(始终采用Continent/City
格式,如America/Sao_Paulo
或Europe/Berlin
)。
避免使用3个字母的缩写(例如IST
或EST
),因为它们是ambiguous and not standard。
要使用其他时区,您可以使用IANA's names之一 - 使用TimeZone.getAvailableIDs()
检查所有可用名称。
旧类(Date
,Calendar
和SimpleDateFormat
)有lots of problems和design issues,并且它们被新API取代
如果您使用 Java 8 ,请考虑使用new java.time API。它更容易,less bugged and less error-prone than the old APIs。
如果您正在使用 Java< = 7 ,则可以使用ThreeTen Backport,这是Java 8新日期/时间类的绝佳后端。对于 Android ,ThreeTenABP(更多关于如何使用它here)。
虽然您也可以使用Joda-Time,但它处于维护模式并且正在被新API取代,因此我不建议使用它来启动新项目。即使在joda's website它也说:"请注意,Joda-Time被认为是一个很大程度上“完成”的项目。没有计划重大改进。如果使用Java SE 8,请迁移到java.time(JSR-310)。" 。
以下代码适用于两者。
唯一的区别是包名称(在Java 8中为java.time
而在ThreeTen Backport(或Android的ThreeTenABP中)为org.threeten.bp
),但类和方法名称是一样的。
获得millis值后,创建日期和转换为某个时区的代码非常相似:
ZoneId zone = ZoneId.of("Asia/Kolkata");
ZonedDateTime z = Instant.ofEpochMilli(1499101493296L).atZone(zone);
System.out.println(z); // 2017-07-03T22:34:53.296+05:30[Asia/Kolkata]
输出将是:
2017-07-03T22:34:53.296 + 05:30 [亚/加尔各答]
如果您想要其他格式,请使用DateTimeFormatter
:
DateTimeFormatter fmt = DateTimeFormatter.ofPattern("dd/MM/yyyy HH:mm:ss x");
System.out.println(z.format(fmt)); // 03/07/2017 22:34:53 +0530
输出将是:
03/07/2017 22:34:53 +0530
如果您需要其他格式,请check the javadoc了解更多详情。
要使用其他时区,您可以使用IANA's names之一 - 使用ZoneId.getAvailableZoneIds()
检查所有可用名称。