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

Showing Local Time based on client / server offset in java

我的 client/browser 在印度,我从​​ javascript 获取时区偏移量 使用以下代码:

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(); 

但是我得到的date/time比预计时间提前了1小时。我错过了什么?

使用 Java SE

进行日期和时间操作

You can print a list of supported TimeZones by using the following code.

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 对象作为参数,并让该方法单独负责减法。然后我要么打印结果,要么将其作为不同方法的一部分。我没有以这种方式构建 post,因为我想在 post.

中包含所有相关代码

使用 Joda 进行日期和时间操作

This type of manipulation has already been solved in Java. The Joda Time Library is probably your best bet if you are doing a lot of date manipulation. If you are only manipulating time in this one instance then it would be a bit over kill to include the dependency in your runtime.

再次打印出时区。

public void printDateTimeZones() {
   for(String zone : DateTimeZone.getAvailableIDs()) {
      System.out.println(zone);
   }
}

然后您可以 return 使用默认格式的两个 DateTimeZone 之间的周期(差异)的字符串,代码如下。

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 提供了一个格式生成器 class,它允许您指定您喜欢的格式。

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))));
}

一个java.util.Date对象has no timezone information。它只有一个 long 值,即从 1970-01-01T00:00:00Z 开始的毫秒数(也称为 "unix epoch""epoch")。这个值绝对独立于时区(你也可以说 "it's in UTC")。

要将此值转换为另一个时区,您不需要在时区之间进行所有这些数学运算。您只需获取此毫秒值并将其转换为所需的时区。

要从 javascript 中获取值,只需执行以下操作:

var d = new Date();
var millis = d.getTime();

变量millis 将包含距纪元的毫秒数。在我做的测试中,这个值是 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() 检查所有可用名称。


新建JavaDate/TimeAPI

旧的 类(DateCalendarSimpleDateFormat)有 lots of problems and design issues,它们正在被新的 APIs.

如果您正在使用 Java 8,请考虑使用 new java.time API. It's easier, less bugged and less error-prone than the old APIs.

如果您使用 Java <= 7,您可以使用 ThreeTen Backport, a great backport for Java 8's new date/time classes. And for Android, there's the ThreeTenABP (more on how to use it ).

虽然你也可以使用 Joda-Time,但它处于维护模式,正在被新的 APIs 取代,所以我不建议用它开始一个新项目.即使在 joda's website 它说:"Note that Joda-Time is considered to be a largely “finished” project. No major enhancements are planned. If using Java SE 8, please migrate to java.time (JSR-310).".

下面的代码适用于两者。 唯一的区别是包名称(在 Java 8 中是 java.time,在 ThreeTen Backport(或 Android 的 ThreeTenABP)中是 org.threeten.bp),但是 类和方法名称相同。

获得毫秒值后,创建日期和转换为某个时区的代码非常相似:

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[Asia/Kolkata]

如果您想要不同的格式,请使用 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() 检查所有可用名称。