假设我的服务器位于美国,我住在俄罗斯。我们知道他们有不同的时区。
我的应用程序从服务器获取文本(String
)。此文本数据在数据库中有Date
列以保留记录日期。
当我获得数据时,我也获得date
知识。所以我可以按时间分组。顶部的第一个和底部的最后一个。无论...
我写了一个函数来显示日期值更具人性化的可读性,例如“13小时”,“9分钟”。服务器在服务器(美国)时区发送日期。
当我计算申请时间与俄罗斯时区(因为它是申请时的当前时区)时,它计算错误。所以,这不是任何人想要的东西。
如何才能实现正确的计算?
注释:此应用程序将由不同国家/地区的公民使用。所以我无法使计算成为静态。
答案 0 :(得分:6)
没有“俄罗斯时区”或“美国时区”这样的东西。这两个国家都有几个不同的时区。请参阅Time in Russia和Time in the USA上的维基百科文章。
您应该始终编写服务器代码,使其不依赖于运行服务器的时区。通常,这是通过将所有时间存储为UTC来完成的。由于客户端是Android设备,只需在客户端上进行本地时间的转换,只需向服务器发送UTC即可。
如果您使用的是Java,则应该使用Joda Time进行转换。它比Calendar
类更清晰,更易于使用。
<强>更新强>
正如Basil在评论中指出的那样,Joda-Time项目现在处于维护模式。该团队建议迁移到JSR 310定义的java.time
类。对于早期的Android,请参阅ThreeTen-Backport和ThreeTenABP项目。请参阅How to use…。
答案 1 :(得分:2)
Calendar类可以在时区之间转换时间,只要您知道服务器和客户端使用的时区即可。如果您移动服务器以避免将来出现问题,最好定义数据库中的时间。我更喜欢使用UTC作为其标准,但您可以使用您希望的任何时区,只要它已定义并在您的文档中,以便您将来知道。
以下是一个显示如何操作的问题:Date and time conversion to some other Timezone in java
答案 2 :(得分:1)
时区与经过的小时 - 分钟 - 秒无关。
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
和&amp; SimpleDateFormat
现在位于Joda-Time的maintenance mode项目建议迁移到java.time类。
要了解详情,请参阅Oracle Tutorial。并搜索Stack Overflow以获取许多示例和解释。规范是JSR 310。
从哪里获取java.time类?
答案 3 :(得分:0)
我不知道如何以程序化的方式实现这一点,但我知道可以做到:) ... 也许您应该尝试一种策略,根据GMT时间保存所有记录,然后在客户端制作一个公式,检查客户的时区,然后如果它是负时区替换当前时间记录的数量和如果它是一个积极的时区将这个数量添加到时间记录这样它将根据时区显示给客户.....(原谅我的语法:)希望它可以)
答案 4 :(得分:0)
如果您在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