用于计算小时差的断言失败

时间:2015-04-28 09:30:21

标签: java date calendar assertion

我从“日历日期”中减去500 hours,而我的方法在数小时内有所不同。

为什么以下断言失败?

测试方法:

 public long dataFetchTotalTime(long time) {
        long hours = 0;
        Calendar ackCal = Calendar.getInstance();
        long diff = ackCal.getTime().getTime() - time;
        hours = TimeUnit.HOURS.convert(diff, TimeUnit.MILLISECONDS);
        return hours;
    }

测试用例:

@Test
public void checkFetchTimeDiff()
{
    long expected = 500;
    Client client = new Client();
    Calendar ackCal = Calendar.getInstance();
    ackCal.set(Calendar.HOUR_OF_DAY, -500);
    long actual = client.dataFetchTotalTime(ackCal.getTime().getTime());
    Assert.assertEquals(expected, actual);
}

错误追踪:

java.lang.AssertionError: expected:<500> but was:<514>
    at org.junit.Assert.fail(Assert.java:88)
    at org.junit.Assert.failNotEquals(Assert.java:834)
    at org.junit.Assert.assertEquals(Assert.java:645)
    at org.junit.Assert.assertEquals(Assert.java:631)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)

2 个答案:

答案 0 :(得分:0)

这是一个错误:

ackCal.set(Calendar.HOUR_OF_DAY, -500);

测试方法:

 public long dataFetchTotalTime(long time) {
        long hours = 0;
        Calendar ackCal = Calendar.getInstance();
        long diff = ackCal.getTime().getTime() - time;
        hours = TimeUnit.HOURS.convert(diff, TimeUnit.MILLISECONDS);
        return hours;
    }

测试用例:

@Test
public void checkFetchTimeDiff()
{
    long expected = 500;
    Client client = new Client();
    Calendar ackCal = Calendar.getInstance();
    ackCal.add(Calendar.HOUR_OF_DAY, -500);
    long actual = client.dataFetchTotalTime(ackCal.getTime().getTime());
    Assert.assertEquals(expected, actual);
}

答案 1 :(得分:0)

tl; dr

ZonedDateTime now = ZonedDateTime.now( ZoneId.systemDefault() ) ;
System.out.println(
    Duration
    .between(
        now , 
        now.plusHours( 500L ) 
    )
    .toHours()
) ;

请参阅此code run live at IdeOne.com

  

500

java.time

您使用的是可怕的日期时间类,而这些类在几年前已被JSR 310中定义的现代 java.time 类取代。

ZonedDateTime

Calendar类,实际上是其常用的子类GregorianCalendar,被ZonedDateTime取代。 ZonedDateTime类代表了某个特定区域的人们通过挂钟时间所看到的时刻。换句话说,是带有时区的日期和时间。

您可以根据需要进行转换,但最好完全避免使用旧的日期时间类。要进行转换,请调用添加到旧类中的新方法。

GregorianCalendar gc = ( GregorianCalendar ) myJavaUtilCalendar  ;  // Cast from the more general to the concrete.
ZonedDateTime zdt = gc.toZonedDateTime() ;  // Convert from legacy class to modern class.

要捕获当前时刻,请指定所需的时区。在任何给定时刻,日期都会在全球范围内变化。

ZoneId z = ZoneId.of( "America/Montreal" ) ;
ZonedDateTime zdt = ZonedDateTime.now( z ) ;

Duration

做你的数学。将未附加到时间轴上的时间跨度表示为Duration(以时-分-秒为单位),或Period(以年-月-日-日为单位)。

Duration d = Duration.ofHours( 500 ) ;
ZonedDateTime zdtLater = zdt.plus( d ) ; 

要确定经过的时间,请使用Duration.between

Duration elapsed = Duration.between( zdt , zdtLater ) ;

要以标准ISO 8601格式查看文本,请致电toString

String output = elapsed.toString() ;

要查看经过的总小时数,请致电Duration::toHours

long hours = elapsed.toHours() ;

Table of date-time types in Java, both modern and legacy.


关于 java.time

java.time框架已内置在Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.DateCalendarSimpleDateFormat

要了解更多信息,请参见Oracle Tutorial。并在Stack Overflow中搜索许多示例和说明。规格为JSR 310

目前位于Joda-Timemaintenance mode项目建议迁移到java.time类。

您可以直接与数据库交换 java.time 对象。使用符合JDBC driver或更高版本的JDBC 4.2。不需要字符串,不需要java.sql.*类。

在哪里获取java.time类?

ThreeTen-Extra项目使用其他类扩展了java.time。该项目为将来可能在java.time中添加内容提供了一个试验场。您可能会在这里找到一些有用的类,例如IntervalYearWeekYearQuartermore