假设我的服务器位于美国,我住在俄罗斯。我们知道他们有不同的时区。
我的应用程序从服务器获取文本(String
)。而这个文本数据在数据库中有Date
列以保存记录日期。
当我获得数据时,我也获得了date
的知识。所以我可以按时间分组。顶部的第一个和底部的最后一个。随你...
我写了一个函数来显示日期值更人性化,如“13小时”,“9分钟”。服务器在服务器(美国)时区发送日期。
当我用俄罗斯时区计算申请时间时(因为它是申请时的当前时区),它计算错误。所以,这不是任何人想要的东西。
我该怎么做才能实现正确的计算?
注释:此应用程序将由不同国家的公民使用。所以我无法使计算静态。
没有“俄罗斯时区”或“美国时区”这样的东西。这两个国家都有几个不同的时区。请参阅有关Time in Russia和Time in the USA的维基百科文章。
您应该始终编写服务器代码,使其不依赖于服务器运行的时区。通常这是通过将所有时间存储为UTC来完成的。由于客户端是Android设备,只需在客户端上转换为本地时间,从服务器发送UTC或从服务器发送UTC。
如果您使用的是Java,那么您应该使用Joda Time进行转换。它比Calendar
类更清洁,更容易使用。
更新
正如Basil在评论中指出的那样,Joda-Time项目现在处于维护模式。该团队建议迁移到java.time
定义的JSR 310类。对于早期的Android,请参阅ThreeTen-Backport和ThreeTenABP项目。见How to use…。
Calendar类可以在时区之间转换时间,只要您知道服务器和客户端使用的时区即可。如果您移动服务器以避免将来出现问题,最好定义数据库中的时间。我更喜欢使用UTC作为其标准,但您可以使用您希望的任何时区,只要它已定义并在您的文档中,以便您将来知道。
这是一个显示如何做的问题:Date and time conversion to some other Timezone in java
时区与经过的小时 - 分钟 - 秒无关。
Duration.between(
Instant.parse( "2017-12-14T01:34:56.123456789Z" ) ;
Instant.now()
)
现代方法使用业界领先的java.time类。这些取代了麻烦的旧遗留日期时间类,如Date
和Calendar
。
通常应将服务器设置为UTC的时区。你永远不应该依赖任何这样的设置。相反,您的代码应指定所需/预期的时区。
作为一般规则,您的业务逻辑,数据存储和数据交换都应该是UTC。序列化为文本时,仅使用标准ISO 8601格式。
必要时应用其他时区,如用户在用户界面中所期望的那样。因此,通常您应该将UTC以外的时区视为本地化问题。
Instant
Instant
类表示一个时刻,UTC时间轴上的一个点,分辨率为纳秒。获取当前时刻不受时区设置的影响。
Instant instant = Instant.now() ;
通过调用toString
以标准格式序列化为文本。
String output = instant.toString() ;
您可以将Instant
与您的数据库进行交换。
myPreparedStatement.setObject( … , instant ) ;
…和…
Instant instant = myPreparedStatement.getObject( … , Instant.class ) ;
ZonedDateTime
使用ZoneId
来获得ZonedDateTime
。
ZoneId z = ZoneId.of( "America/Montreal" ) ;
ZonedDateTime zdt = instant.atZone( z ) ;
要以非标准格式生成字符串,请使用DateTimeFormatter
类。搜索Stack Overflow以获取许多示例和讨论。
Duration
要获得经过的小时数,分钟数,秒数,请使用Duration
类。请注意,您不需要经过时间的时区; UTC(Instant
)给出了相同的结果。
Duration d = Duration.between( then , Instant.now() ) ;
对于早期的Android,请参阅ThreeTen-Backport和ThreeTenABP项目。见How to use…。
java.time框架内置于Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,如java.util.Date
,Calendar
和SimpleDateFormat
。
现在在Joda-Time的maintenance mode项目建议迁移到java.time班。
要了解更多信息,请参阅Oracle Tutorial。并搜索Stack Overflow以获取许多示例和解释。规格是JSR 310。
从哪里获取java.time类?
我不知道如何以程序化方式实现这一目标,但我知道可以做到:) ...也许您应该尝试一种策略,根据GMT时间保存所有记录,然后在客户端制作公式它检查客户的时区,然后如果它是负时区替换当前时间记录的金额,如果是正时区,将这个金额添加到时间记录这样它将根据时间显示给客户有时区.....(原谅我的语法:)祝它可以)
如果您在Java 8之前使用Java,那么您应该使用Joda Time进行转换。看看这个问题:Date and time conversion to some other Timezone in java
如果您正在使用Java 8及更高版本(或者如果您计划在公司中启动升级过程),Java 8将包含一个正确的日期/时间库,并且Joda的创建者建议切换到使用它。看看这个问题:Joda Time - Convert UTC DateTime to Date - Using a Version 1.2.1.1 of Joda