根据 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_Paulo
或 Europe/Berlin
)。
避免使用 3 个字母的缩写(如 IST
或 EST
),因为它们是 ambiguous and not standard.
要使用另一个时区,您可以使用 IANA's names - 使用 TimeZone.getAvailableIDs()
检查所有可用名称。
新建JavaDate/TimeAPI
旧的 类(Date
、Calendar
和 SimpleDateFormat
)有 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()
检查所有可用名称。
我的 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_Paulo
或 Europe/Berlin
)。
避免使用 3 个字母的缩写(如 IST
或 EST
),因为它们是 ambiguous and not standard.
要使用另一个时区,您可以使用 IANA's names - 使用 TimeZone.getAvailableIDs()
检查所有可用名称。
新建JavaDate/TimeAPI
旧的 类(Date
、Calendar
和 SimpleDateFormat
)有 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()
检查所有可用名称。