java中算两个时间内的秒数_java – Joda Time – 计算两个日期之间的秒数会引发异常....

我使用以下代码来计算两个日期之间的秒数差异:

long secondsBetween = (Seconds.secondsBetween(new LocalDate("1901-01-01"), new LocalDate()).getSeconds());

但是我得到以下异常:

08-08 18:21:27.345: E/AndroidRuntime(6972): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.testbdr/com.testbdr.MainActivity}: java.lang.ArithmeticException: Value cannot fit in an int: 3584908800

08-08 18:21:27.345: E/AndroidRuntime(6972): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2189)

08-08 18:21:27.345: E/AndroidRuntime(6972): at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2216)

08-08 18:21:27.345: E/AndroidRuntime(6972): at android.app.ActivityThread.access$600(ActivityThread.java:149)

08-08 18:21:27.345: E/AndroidRuntime(6972): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1305)

08-08 18:21:27.345: E/AndroidRuntime(6972): at android.os.Handler.dispatchMessage(Handler.java:99)

08-08 18:21:27.345: E/AndroidRuntime(6972): at android.os.Looper.loop(Looper.java:153)

08-08 18:21:27.345: E/AndroidRuntime(6972): at android.app.ActivityThread.main(ActivityThread.java:5000)

08-08 18:21:27.345: E/AndroidRuntime(6972): at java.lang.reflect.Method.invokeNative(Native Method)

08-08 18:21:27.345: E/AndroidRuntime(6972): at java.lang.reflect.Method.invoke(Method.java:511)

08-08 18:21:27.345: E/AndroidRuntime(6972): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:821)

08-08 18:21:27.345: E/AndroidRuntime(6972): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:584)

08-08 18:21:27.345: E/AndroidRuntime(6972): at dalvik.system.NativeStart.main(Native Method)

08-08 18:21:27.345: E/AndroidRuntime(6972): Caused by: java.lang.ArithmeticException: Value cannot fit in an int: 3584908800

08-08 18:21:27.345: E/AndroidRuntime(6972): at org.joda.time.field.FieldUtils.safeToInt(FieldUtils.java:206)

08-08 18:21:27.345: E/AndroidRuntime(6972): at org.joda.time.field.BaseDurationField.getDifference(BaseDurationField.java:141)

08-08 18:21:27.345: E/AndroidRuntime(6972): at org.joda.time.chrono.BaseChronology.get(BaseChronology.java:260)

08-08 18:21:27.345: E/AndroidRuntime(6972): at org.joda.time.base.BaseSingleFieldPeriod.between(BaseSingleFieldPeriod.java:105)

08-08 18:21:27.345: E/AndroidRuntime(6972): at org.joda.time.Seconds.secondsBetween(Seconds.java:124)

08-08 18:21:27.345: E/AndroidRuntime(6972): at com.testbdr.MainActivity.onCreate(MainActivity.java:27)

08-08 18:21:27.345: E/AndroidRuntime(6972): at android.app.Activity.performCreate(Activity.java:5020)

08-08 18:21:27.345: E/AndroidRuntime(6972): at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1080)

08-08 18:21:27.345: E/AndroidRuntime(6972): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2153)

08-08 18:21:27.345: E/AndroidRuntime(6972): ... 11 more

解决方法:

INT

正如其他答案正确陈述的那样,问题是你和Joda-Time正在使用int来处理秒数.一个32位的int只能容纳大约68年的秒数.

如果你坚持使用秒来跟踪几个世纪的时间,你必须使用64位长而不是32位int.

顺便说一下,在Unix中使用32位int来跟踪时间的秒数表示现实世界的问题知道Year 2038 problem.

秒不用于长时间跨度

正如其他人所说,使用秒来跟踪如此长的时间跨度是不寻常的.如果可能的话,您可能想重新考虑这个前提.

一种替代方案:ISO 8601标准提供Durations格式的PnYnMnDTnHnMnS,包括年,月,日等. Joda-Time知道如何解析并生成这样的字符串(Period和Duration类).虽然Joda-Time只能处理数字秒数,但是当它以ISO 8601格式呈现为字符串时,它可以处理更大的秒数,如下面的代码示例所示(PT3584908800S).

毫秒

Joda-Time使用毫秒内部跟踪epoch的计数. Joda-Time提供了访问这些毫秒长值的方法.

我通常建议你在几毫秒内完成日期工作.但在你的情况下,它是有道理的,根据需要转换为秒.

一天的开始

要计算毫秒数,我们需要使用DateTime而不是LocalDate.

养成用TimeAtStartOfDay调用方法的习惯,以获得当天的第一个时刻.这个时间通常是00:00:00,但并不总是因为夏令时或其他异常.

时区

即使对于LocalDate,时区也是至关重要的.日期(和一天的第一时刻)由时区决定.巴黎的新日早些时候比蒙特利尔更新.

如果省略时区,将使用JVM的当前默认时区.通常更好地明确并指定所需的时区.我怀疑你的目的,使用UTC是有道理的.

持续时间

Joda-Time提供了Duration课程,表示与时间轴(宇宙历史)无关的时间跨度.

示例代码

使用Joda-Time 2.4的示例代码.

DateTime history = new DateTime( "1901-01-01", DateTimeZone.UTC ).withTimeAtStartOfDay(); // Technically, the call to withTimeAtStartOfDay is not necessary here as Joda-Time defaults to that for parsing a date-only string. But the call is a good habit and makes clear out intention.

DateTime today = new DateTime( DateTimeZone.UTC ).withTimeAtStartOfDay();

Duration duration = new Duration( history, today );

long millis = duration.getMillis(); // Use a long, not an int.

long seconds = ( millis / 1000L ); // Use a long, not an int. Maybe use BigDecimal or BigInteger if you want rounding.

转储到控制台.

System.out.println( "history: " + history );

System.out.println( "today: " + today );

System.out.println( "duration: " + duration );

System.out.println( "millis: " + millis );

System.out.println( "seconds: " + seconds );

跑步时

history: 1901-01-01T00:00:00.000Z

today: 2014-08-08T00:00:00.000Z

duration: PT3584908800S

millis: 3584908800000

seconds: 3584908800

当走向另一个方向时,要么:

标签:android,java,datetime,jodatime

来源: https://codeday.me/bug/20190722/1506364.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值