关于根据字段排序的问题

最近在项目中遇见一个需要根据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的场合。

你可能感兴趣的:(排序)