最近在项目中遇见一个需要根据java类的字段排序的问题。为什么不直接在数据库里面进行排序呢?因为那是一张比较复杂的报表,显示的数据没什么规律,所以当时采用的是先查询出来然后再组装成一个java对象列表,现在领导居然要求根据列名排序!没办法,只得去试试。一般这种问题会有两种解决办法:
1.自己写个算法对列表的对象进行排序
2.实现对象的compareTo方法,然后用jdk自带的排序功能
个人比较倾向于第二种方法,但是起初想,要列的不同compareTo的实现也会不同,那么那个compareTo方法岂不是要动态生成?于是自己试着去看看cglib这个库,但这个库没什么文档,在网上的资料上看,主要也是用在动态代理上,能不能做到还难说,就算能,我还得花时间去看源码,于是放弃了。当然放弃之前经一同事提醒,可以采用一种迂回的方法,那就是定义一个排序字段,根据这个字段的值来判断对哪一列进行排序。如:
private String used0 = new String("0");
private String used1 = new String("0");
private String used2 = new String("0");
private String used4 = new String("0");
//排序字段,用于compareTo
private static String sortInput = "";
即上面的sortInput字段,如果sortInput="used0",那么就根据used0进行排序。具体实现如下:
1.实现Comparable接口,最后加上类型,如Comparable<McRegData>
2.实现compareTo方法:
@Override
public int compareTo(McRegData m) {
if(sortInput.equals("used0")){
return new BigDecimal(used0).compareTo(new BigDecimal(m.getUsed0()));
}else if(sortInput.equals("used1")){
return new BigDecimal(used1).compareTo(new BigDecimal(m.getUsed1()));
}else if(sortInput.equals("used2")){
return new BigDecimal(used2).compareTo(new BigDecimal(m.getUsed2()));
}else if(sortInput.equals("used4")){
return new BigDecimal(used4).compareTo(new BigDecimal(m.getUsed4()));
}else if(sortInput.equals("usedE")){
return new BigDecimal(usedE).compareTo(new BigDecimal(m.getUsedE()));
}else if(sortInput.equals("usedU")){
return new BigDecimal(usedU).compareTo(new BigDecimal(m.getUsedU()));
}else if(sortInput.equals("used3")){
return new BigDecimal(used3).compareTo(new BigDecimal(m.getUsed3()));
}else if(sortInput.equals("usedR")){
return new BigDecimal(usedR).compareTo(new BigDecimal(m.getUsedR()));
}else if(sortInput.equals("total")){
return new BigDecimal(this.getTotal()).compareTo(new BigDecimal(m.getTotal()));
}
return 0;
因为只比较数字字段,所以把它转型成BigDecimal,上面只是对临时组装的对象类进行了相应的处理,最后再对整个列表数据进行排序:
if(map.get("sortInput")!=null&&map.get("sortType")!=null){
String sortInput = map.get("sortInput").toString();
String sortType = map.get("sortType").toString();
//设置需要排序的字段
McRegData.setSortInput(sortInput);
Collections.sort(regList);
if(sortType.equals("desc")){
Collections.reverse(regList);
}
}
其中map.get("sortInput")等是外部传进来的排序参数。上面的代码中,对于没有纳入排序的列也可以实现把数据倒转过来的效果(Collecions.reverse实现)。不过有一个地方还可简化下,那就是那个看起来比较长的compareTo方法,只要传入的排序列与java对象的属性列名相同,就可以通过反射的方式来简化,如:
public int compareTo(McRegData m) {
try {
BigDecimal old_field = new BigDecimal(MethodUtils.invokeMethod(this, "get"+StringUtils.capitalize(sortInput)).toString());
BigDecimal new_field = new BigDecimal(MethodUtils.invokeMethod(m, "get"+StringUtils.capitalize(sortInput)).toString());
return old_field.compareTo(new_field);
} catch (Exception e) {
//非数字列会出现异常
e.printStackTrace();
}
return 0;
}
上面的代码用到了两个commons.lang下的工具类,不过代码相对以前的来说,简化了不少,而且不受字段多少影响,看起来舒服多了。每次解决了棘手的问题后总会有不少的收获,虽然过程比较痛苦,毕竟项目是需要进度的,谁敢顶着压力来研究这些呢,只能在有空的时候才来优化。
(2012.1.29):最近偶尔想起,这是一种典型的该使用Comparator而不是Comparable的场合。