根据java中的客户端/服务器偏移显示本地时间

时间:2017-07-03 15:13:22

标签: java calendar timezone client gmt

我的客户端/浏览器在印度,我从​​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小时。我错过了什么?

2 个答案:

答案 0 :(得分:3)

使用Java SE进行日期和时间操作

  

您可以使用以下代码打印支持的时区列表。

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对象作为参数传递,并保持该方法单独负责减法。然后我会打印结果或使其成为不同方法的一部分。我没有以这种方式构建帖子,因为我想在帖子中包含所有相关代码。

使用Joda

操作日期和时间
  

这种类型的操作已经用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_PauloEurope/Berlin)。 避免使用3个字母的缩写(例如ISTEST),因为它们是ambiguous and not standard

要使用其他时区,您可以使用IANA's names之一 - 使用TimeZone.getAvailableIDs()检查所有可用名称。

新Java日期/时间API

旧类(DateCalendarSimpleDateFormat)有lots of problemsdesign 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()检查所有可用名称。