我有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]]
怎么了?
答案 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]]
您可以扩展代码以检查小时是否在预期的时间间隔内,例如,如果没有,则抛出异常。