java – 使用JodaTime以毫秒为单位的UTC到本地时间
我试图使用Jodatime显示特定时间段内的交易.
我们的服务器要求开始日期和结束日期为UTC(这可能是显而易见的).因此,围绕它们的任何业务逻辑都使用DateTime对象,时区设置为DateTimeZone.UTC,例如, mStartDate = DateTime.now(UTC).withTimeAtStartOfDay(); 除了显示我不知道如何为本地(系统默认)时区增加它的时间之外,这很有效.理想情况下,我想使用传递两个本地时间戳的DateUtils formatDateRange函数.但getMillis()函数似乎不考虑本地偏移: 我也试过这个: mTimePeriodTitle.setText(DateUtils.formatDateRange(mContext,f,mStartDate.getMillis(),mEndDate.getMillis(),DateUtils.FORMAT_SHOW_TIME,TimeZone.getDefault().getID()).toString()); 但它没有任何区别.所以我的问题是如何获得具有2个UTC时间戳的格式良好的本地日期范围? 解决方法
如果您的DateTime是UTC并且您想将其转换为另一个时区,则可以使用withZone方法进行转换.
对于下面的示例,我的默认时区是America / Sao_Paulo(您可以使用DateTimeZone.getDefault()检查您的时区): // create today's date in UTC DateTime mStartDate = DateTime.now(DateTimeZone.UTC).withTimeAtStartOfDay(); // date/time in UTC System.out.println(mStartDate); // 2017-06-13T00:00:00.000Z // date/time in my default timezone (America/Sao_Paulo) System.out.println(mStartDate.withZone(DateTimeZone.getDefault())); // 2017-06-12T21:00:00.000-03:00 输出是:
请注意,withZone方法正确地将日期和时间转换为我的时区(在America / Sao_Paulo中,当前偏移量为UTC-03:00),因此进行了相应调整. 如果您想获得时间(小时/分钟/秒),可以使用toLocalTime()方法: System.out.println(mStartDate.withZone(DateTimeZone.getDefault()).toLocalTime()); // 21:00:00.000 输出是:
如果您想要其他格式(例如,不打印秒数的3位数),则可以使用DateTimeFormatter.好处是您可以在格式化程序中设置时区,因此您无需转换DateTime: // create formatter for hour/minute/second,set it with my default timezone DateTimeFormatter fmt = DateTimeFormat.forPattern("HH:mm:ss").withZone(DateTimeZone.getDefault()); System.out.println(fmt.print(mStartDate)); // 21:00:00 输出是:
要获得范围,可以使用上述方法之一和DateTime(mStartDate和mEndDate),并使用DateTimeFormatter更改为您需要的任何格式. PS:我认为你在使用getMillis()时缺少的是日期时间(UTC和默认时区)代表同一时刻.你只是将这一瞬间转换为当地时间,但毫秒是相同的(想想,此时此刻,世界上的每个人都处于同一时刻(相同的毫秒),但他们的当地时间可能会有所不同取决于他们在哪里).因此,在将UTC DateTime转换为另一个时区时,我们只是找到该区域中的当地时间,该时间对应于相同的millis. 您可以在两个对象上使用getMillis()方法检查: System.out.println(mStartDate.getMillis()); // 1497312000000 System.out.println(mStartDate.withZone(DateTimeZone.getDefault()).getMillis()); // 1497312000000 请注意,即使我将对象转换为另一个时区,millis仍保持不变(1497312000000).那是因为两者都代表同一时刻,我只是将它们移动到另一个时区,其中各自的当地时间不同. Java新的日期/时间API Joda-Time它被停用并被新的API取代,所以我不建议用它开始一个新项目.如果是这种情况,您可以考虑使用新的日期/时间API,但如果您使用Joda的代码库很大或者现在不想迁移它,您可以考虑其余的答案. 无论如何,即使在joda’s website它说:“请注意,Joda-Time被认为是一个很大程度上已完成的项目.没有计划进行重大改进.如果使用Java SE 8,请迁移到java.time(JSR-310).” *. 如果您正在使用Java 8,请考虑使用new java.time API.它更容易,less bugged and less error-prone than the old APIs.我不确定它是否已经可用于所有Android版本(但请参阅下面的替代方案). 如果您使用Java< = 7,则可以使用ThreeTen Backport,这是Java 8的新日期/时间类的一个很好的后端.对于Android,有一种方法可以使用它,使用ThreeTenABP(更多关于如何使用它here). 以下代码适用于两者. 要以UTC开头获取当前日期,您可以执行以下操作: // UTC's today at start of the day ZonedDateTime utc = LocalDate.now(ZoneOffset.UTC).atStartOfDay(ZoneOffset.UTC); System.out.println(utc); // 2017-06-13T00:00Z 首先,我使用LocalDate.now(ZoneOffset.UTC)来查找UTC中的当前本地日期.如果我只使用LocalDate.now(),它将在我的默认时区得到当前日期,这不是我们想要的(它可能与UTC不同,具体取决于你在哪里 – 何时 – 你是什么以及默认时区是什么是). 然后我使用atStartOfDay(ZoneOffset.UTC)来获取UTC当天的开始.我知道使用UTC两次听起来多余,但是API允许我们在这种方法中使用任何时区,而IMO它明确了我们想要的时区(如果日期是在夏令时变化的时区,那么一天的开始可能不是午夜 – 时区参数是为了保证设置正确的值). 输出是:
要转换为我的默认时区,我可以使用ZoneId.systemDefault(),在我的例子中返回America / Sao_Paulo.要转换它并只获取本地时间部分,只需执行以下操作: System.out.println(utc.withZoneSameInstant(ZoneId.systemDefault()).toLocalTime()); // 21:00 输出是:
如果要更改它,还可以使用格式化程序: // formatter for localtime (hour/minute/second) DateTimeFormatter fmt = DateTimeFormatter.ofPattern("HH:mm:ss"); System.out.println(fmt.format(utc.withZoneSameInstant(ZoneId.systemDefault()))); // 21:00:00 输出是:
(编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |