为什么java Date比较在00:00 AM时返回false?

时间:2016-08-24 17:30:04

标签: java android date

我正在将日期对象与两个日期对象进行比较。我想知道对象在该范围内。所以这是我要比较的代码: 我会得到三个日期对象,一个属于我当前的时间,另外两个属于前后时间。

  public String calcOpenClosed(String timeRange, Date currentTime)
{


    Log.e("current time got", currentTime.getHours()+" "+currentTime.getMinutes());
    if(timeRange.contains(","))
    {

        String[] slot = timeRange.split(",");
        String[] range1 = slot[0].split("-");
        String[] range2 = slot[1].split("-");
        Date date1;
        Date date2;
        Date date3;
        Date date4;
        Log.e("range 1", range1[0]);
        Log.e("range 2", range1[1]);
        Log.e("range 3", range2[0]);
        Log.e("range 4", range2[1]);

        date1 = getTime(range1[0]);
        date2 = getTime(range1[1]);
        date3 = getTime(range2[0]);
        date4 = getTime(range2[1]);

        Calendar cal1 = Calendar.getInstance();
        Calendar cal2 = Calendar.getInstance();
        Calendar cal3 = Calendar.getInstance();
        Calendar cal4 = Calendar.getInstance();
        cal1.setTime(date1);
        cal2.setTime(date2);
        cal3.setTime(date3);
        cal4.setTime(date4);
        Calendar curr = Calendar.getInstance();
        curr.setTime(currentTime);

        if(curr.after(cal1) && curr.before(cal2) || curr.after(cal3) && curr.after(cal4))
        {
            return "OPEN NOW";
        }
        else
        {
            return "CLOSED";
        }
    }
    else
    {

        String[] range = timeRange.split("-");
        Date date1;
        Date date2;
        date1 = getTime(range[0]);
        date2 = getTime(range[1]);

        Calendar cal1 = Calendar.getInstance();
        Calendar cal2 = Calendar.getInstance();

        cal1.setTime(date1);
        cal2.setTime(date2);

        Calendar curr = Calendar.getInstance();
        curr.setTime(currentTime);

        if(curr.after(cal1) || curr.before(cal2))
        {
            return "OPEN NOW";
        }
        else
        {
            return "CLOSED";
        }
    }
}

public Date getTime(String timeCreated) {

    Calendar calendar = Calendar.getInstance();
    int day = calendar.get(Calendar.DAY_OF_MONTH);
    int month = calendar.get(Calendar.MONTH)+1;
    int year = calendar.get(Calendar.YEAR);
    String date = year + "-" +(month<10?("0"+month):(month)) +"-"+day;
    String timeCreatedSlot[] = timeCreated.split(" ");
    String[] splittedTime = timeCreatedSlot[0].split(":");
    int time = Integer.parseInt(splittedTime[0]);
    String timeNow = String.valueOf(time<10?("0"+time):(time));
    String finalTime = date + " "+ timeNow + ":" + splittedTime[1]+  " "+timeCreatedSlot[1];
    SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd hh:mm aaa");

    try {
        Date timeCreatedDate = dateFormat.parse(finalTime);
        Log.e("time created", timeCreatedDate.getHours()+"");
        return timeCreatedDate;

    } catch (ParseException e) {
        Log.e("exception", "setTimestamp: " + Log.getStackTraceString(e));

    }
    return null;
}

它的工作正常但是当时间范围是凌晨4:00 - 00:00时它给我错误的结果。上面的代码我尝试过date.before和date.after。但给出相同的结果。

1 个答案:

答案 0 :(得分:0)

首先,将String处理与日期时间代码分开。将字符串解析为日期时间对象,然后将这些对象移交给单独的逻辑方法。如果您避免过度依赖字符串而不是绕过物体,那么您将使您的生活更轻松。

其次,您正在使用现在由java.time类取代的麻烦的旧遗留日期时间类。

如果您尝试将开 - 关时间作为通用时间值而不是特定日期 - 时刻,请使用LocalTime类。

LocalTime openingTime = LocalTime.parse ( "04:00" );
LocalTime closingTime = LocalTime.parse ( "00:00" );
Boolean inRange = null;
LocalTime target = LocalTime.MIDNIGHT; // LocalTime.of ( 3 , 0) or LocalTime.of ( 4 , 0) or LocalTime.of ( 5 , 0) or LocalTime.now ( ZoneId.of ( "America/Los_Angeles" ) );
if ( closingTime.equals ( LocalTime.MIDNIGHT ) ) {
    inRange = (  ! target.isBefore ( openingTime ) );
} else {
    inRange = ( (  ! target.isBefore ( openingTime ) ) && ( target.isBefore ( closingTime ) ) );
}

或者您可以在一条线上编写该逻辑。我个人会坚持使用if-else结构。

Boolean inRange2 = ( (  ! target.isBefore ( openingTime ) ) && ( closingTime.equals ( LocalTime.MIDNIGHT ) ? true : target.isBefore ( closingTime ) ) );

转储到控制台。

System.out.println ( "openingTime: " + openingTime + " | closingTime: " + closingTime + " | target: " + target + " | inRange: " + inRange + " | inRange2: " + inRange2 );

在实际工作中,我会通过检查开头后的结束时间来验证输入(或等于LocalTime.MIDNIGHT作为唯一的例外)。

半开

上面的代码采用Half-Open方法将一段时间解释为 inclusive ,而结尾是 exclusive 。这是日期工作中合理的常见做法。

关于java.time

java.time框架内置于Java 8及更高版本中。这些类取代了旧的麻烦日期时间类,例如java.util.Date.Calendar和&amp; java.text.SimpleDateFormat

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

要了解详情,请参阅Oracle Tutorial。并搜索Stack Overflow以获取许多示例和解释。

大部分java.time功能都被反向移植到Java 6&amp; ThreeTen-Backport中的7,并进一步适应Android中的ThreeTenABP(见How to use…)。

ThreeTen-Extra项目使用其他类扩展java.time。该项目是未来可能添加到java.time的试验场。您可以在此处找到一些有用的课程,例如IntervalYearWeekYearQuarter等。