如何在java中对Date进行排序

时间:2017-04-28 11:22:59

标签: java sorting datetime

我有List AttendedExamName对象

public class AttendedExamName {
    private String name;
    private String dateTime;

  //getters    
  //settrs    
}

当尝试使用下面的代码

对对象进行排序时,我尝试基于dateTime属性进行排序
Collections.sort(examNames, (n1,n2)-> {
            try {
                Date date1= new SimpleDateFormat("yyyy-MM-dd-hh.mm.ss").parse(n1.getDateTime());
                Date date2=new SimpleDateFormat("yyyy-MM-dd-hh.mm.ss").parse(n2.getDateTime());
                return date1.compareTo(date2);
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
                return 0;
            }           
            });

我得到了这个结果:

[[name=SCJP, dateTime=2017-04-15-04.37.57],
 [name=SCJP, dateTime=2017-04-15-06.31.52], 
 [name=SCJP, dateTime=2017-04-24-02.17.37], 
 [name=SCJP, dateTime=2017-04-26-05.10.29],
 [name=SCJP, dateTime=2017-04-27-03.30.34],
 [name=SCJP, dateTime=2017-04-28-12.11.09],
 [name=SCJP, dateTime=2017-04-28-01.16.59],
 [name=SCJP, dateTime=2017-04-28-02.10.54],
 [name=SCJP, dateTime=2017-04-28-11.31.16],
 [name=SCJP, dateTime=2017-04-28-11.57.56]]

预期:

[[name=SCJP, dateTime=2017-04-15-04.37.57],
 [name=SCJP, dateTime=2017-04-15-06.31.52], 
 [name=SCJP, dateTime=2017-04-24-02.17.37], 
 [name=SCJP, dateTime=2017-04-26-05.10.29],
 [name=SCJP, dateTime=2017-04-27-03.30.34],
 [name=SCJP, dateTime=2017-04-28-11.31.16],
 [name=SCJP, dateTime=2017-04-28-11.57.56],
 [name=SCJP, dateTime=2017-04-28-12.11.09],
 [name=SCJP, dateTime=2017-04-28-01.16.59],
 [name=SCJP, dateTime=2017-04-28-02.10.54]]

怎么了?

2 个答案:

答案 0 :(得分:6)

很好的一个:问题出在这里:

yyyy-MM-dd-hh.mm.ss

预计时间会达到12小时。但是你需要24小时制,所以你的字符串格式应该是:

yyyy-MM-dd-HH.mm.ss

引用javadoc

  

H小时(0-23)

     

h上午/下午(1-12)小时

除此之外:我建议将该Comparator写为给予sort()的lambda表达式。那件事值得拥有自己的内心阶级;或者:在您的班级上实施Comparable界面!并且为了记录:您还希望避免代码重复。写下格式字符串两次已经创建了代码重复;从而有可能产生错误。

当然:为什么首先要在该类中存储字符串?您应该将转换转换为实际日期ONCE并存储数据对象,而不是在排序期间进行数十亿次!

答案 1 :(得分:2)

由于您使用的是lambda,我假设您使用的是Java 8(或更高版本)。在这种情况下,你没有理由想要使用古老的Date类来解决问题。 Java 8中引入的LocalDateTime通常对程序员更友好。

最佳解决方案是,如果您可以将类中的实例变量更改为类型LocalDateTime

private String name;
private LocalDateTime dateTime;

使用getter,在Java 8中按属性排序的方法是:

    Collections.sort(examNames, Comparator.comparing(AttendedExamName::getDateTime));

结果:

[[name=SCJP, dateTime=2017-04-15T04:37:57], 
 [name=SCJP, dateTime=2017-04-15T06:31:51], 
 [name=SCJP, dateTime=2017-04-24T02:17:37], 
 [name=SCJP, dateTime=2017-04-26T05:10:29], 
 [name=SCJP, dateTime=2017-04-27T03:30:34], 
 [name=SCJP, dateTime=2017-04-28T11:31:16], 
 [name=SCJP, dateTime=2017-04-28T11:57:56], 
 [name=SCJP, dateTime=2017-04-28T12:11:09], 
 [name=SCJP, dateTime=2017-04-28T13:16:59], 
 [name=SCJP, dateTime=2017-04-28T14:10:54]]

我不知道我是否正确猜到了您的预期时间,但您可以解决这个问题。如果您需要其中一些,LocalDateTime当然也可以格式化为12小时和可选的AM / PM标记。

如果您无法更改实例变量的类型,该怎么办?我首先想到的是完全相同的代码。如果可以在数字上从01到12对同一日期的小时进行排序,则可以这样。然后您可以将dateTime字符串排序为字符串并获得正确的时间顺序。但是,我怀疑你不想要这个订单,虽然我还没有完全理解你想要的订单。

一次尝试将是:

    DateTimeFormatter format = DateTimeFormatter.ofPattern("uuuu-MM-dd-HH.mm.ss");
    Collections.sort(examNames, 
            Comparator.comparing(name -> LocalDateTime.parse(name.getDateTime(), format)));

结果:

[[name=SCJP, dateTime=2017-04-15-04.37.57], 
 [name=SCJP, dateTime=2017-04-15-06.31.51], 
 [name=SCJP, dateTime=2017-04-24-02.17.37], 
 [name=SCJP, dateTime=2017-04-26-05.10.29], 
 [name=SCJP, dateTime=2017-04-27-03.30.34], 
 [name=SCJP, dateTime=2017-04-28-01.16.59], 
 [name=SCJP, dateTime=2017-04-28-02.10.54], 
 [name=SCJP, dateTime=2017-04-28-11.31.16], 
 [name=SCJP, dateTime=2017-04-28-11.57.56], 
 [name=SCJP, dateTime=2017-04-28-12.11.09]]

这不是你所说的预期顺序。相反,让我假设您的所有时间都在上午11点到下午7点之间,因为这与您的预期订单一致(11和12在1和2之前,4在6之前)。使用LocalDateTime并不是那么糟糕:

    Collections.sort(examNames, Comparator.comparing(name -> {
        LocalDateTime dt = LocalDateTime.parse(name.getDateTime(), format);
        if (dt.getHour() < 9) {
            return dt.plusHours(12);
        } else {
            return dt;
        }
    }));

现在你得到了预期的订单:

[[name=SCJP, dateTime=2017-04-15-04.37.57], 
 [name=SCJP, dateTime=2017-04-15-06.31.51], 
 [name=SCJP, dateTime=2017-04-24-02.17.37], 
 [name=SCJP, dateTime=2017-04-26-05.10.29], 
 [name=SCJP, dateTime=2017-04-27-03.30.34], 
 [name=SCJP, dateTime=2017-04-28-11.31.16], 
 [name=SCJP, dateTime=2017-04-28-11.57.56], 
 [name=SCJP, dateTime=2017-04-28-12.11.09], 
 [name=SCJP, dateTime=2017-04-28-01.16.59], 
 [name=SCJP, dateTime=2017-04-28-02.10.54]]

您可以扩展代码以检查小时是否在预期的时间间隔内,例如,如果没有,则抛出异常。