1. 要求

对列表(List)中的自定义对象,要求能够按照对象的属性(字段)进行排序(正序、倒序)。

如:用户对象(Member)有用户名(username)、级别(level)、出生日期(birthday)等字段,要求可以分别对它的三个字段进行排序。

2. 实现思路

1. 对于自定义对象,可以在自定义对象中实现Comparable接口,然后再调用Collections.sort的方法实现排序,只能是针对一个属性(字段),维持一个顺序;要实多字段任意选择一个排序,同样需要通过调用Collections.sort(List<T> list, Comparator<? super T> c)方法,传进一个Comparator来实现。

2. 为避免上述步骤中复杂且重复的代码,可以写一个通用的排序类,能够对自定义对象,针对不同的属性(字段),实现排序(正序、倒序)。

3. 实现代码

Member类

package com.clzhang.sample.collections;

import java.text.SimpleDateFormat;
import java.util.Date; /**
* 测试用实体类
* @author acer
*
*/
public class Member implements Comparable<Member>{
// 格式化日期用
private static final SimpleDateFormat MY_SDF = new SimpleDateFormat(
"yyyy-MM-dd"); // 几个属性
private int id;
private String username;
private int level;
private Date birthday; // 构造函数
public Member(int id, String username, int level, String birthday) throws Exception {
this.id = id;
this.username = username;
this.level = level;
this.birthday = new Date(MY_SDF.parse(birthday).getTime());
} // Getters
public String getUsername() {
return username;
} public int getLevel() {
return level;
} public Date getBirthday() {
return birthday;
} // 返回打印用
@Override
public String toString() {
return id + "|" + username + "|" + level + "|" + MY_SDF.format(birthday);
} // 注意:如果使用MySortList类,则此方法不再需要。因为此方法是提供给Collections.sort方法使用的。
@Override
public int compareTo(Member m) {
// 只能对一个字段做比较,如果做整个对象的比较就实现不了按指定字段排序了。
return this.getUsername().compareTo(m.getUsername());
}
}

MySortList类

package com.clzhang.sample.collections;

import java.util.*;
import java.lang.reflect.Method;
import java.lang.reflect.InvocationTargetException; /**
* 这是一个自定义排序的类,专门针对列表(List)中的数据进行排序;可按指定方法进行。
* 目前实现对字符串(String)、日期(Date)、整型(Integer)等三种对象进行排序。
* @author acer
*
* @param <E>
*/
public class MySortList<E> { /**
* 对列表中的数据按指定字段进行排序。要求类必须有相关的方法返回字符串、整型、日期等值以进行比较。
* @param list
* @param method
* @param reverseFlag
*/
public void sortByMethod(List<E> list, final String method,
final boolean reverseFlag) {
Collections.sort(list, new Comparator<Object>() {
@SuppressWarnings("unchecked")
public int compare(Object arg1, Object arg2) {
int result = 0;
try {
Method m1 = ((E) arg1).getClass().getMethod(method, null);
Method m2 = ((E) arg2).getClass().getMethod(method, null);
Object obj1 = m1.invoke(((E)arg1), null);
Object obj2 = m2.invoke(((E)arg2), null);
if(obj1 instanceof String) {
// 字符串
result = obj1.toString().compareTo(obj2.toString());
}else if(obj1 instanceof Date) {
// 日期
long l = ((Date)obj1).getTime() - ((Date)obj2).getTime();
if(l > 0) {
result = 1;
}else if(l < 0) {
result = -1;
}else {
result = 0;
}
}else if(obj1 instanceof Integer) {
// 整型(Method的返回参数可以是int的,因为JDK1.5之后,Integer与int可以自动转换了)
result = (Integer)obj1 - (Integer)obj2;
}else {
// 目前尚不支持的对象,直接转换为String,然后比较,后果未知
result = obj1.toString().compareTo(obj2.toString()); System.err.println("MySortList.sortByMethod方法接受到不可识别的对象类型,转换为字符串后比较返回...");
} if (reverseFlag) {
// 倒序
result = -result;
}
} catch (NoSuchMethodException nsme) {
nsme.printStackTrace();
} catch (IllegalAccessException iae) {
iae.printStackTrace();
} catch (InvocationTargetException ite) {
ite.printStackTrace();
} return result;
}
});
} // 测试函数
public static void main(String[] args) throws Exception {
// 生成自定义对象,然后对它按照指定字段排序
List<Member> listMember = new ArrayList<Member>();
listMember.add(new Member(1, "wm123", 3, "1992-12-01"));
listMember.add(new Member(2, "a234", 8, "1995-12-01"));
listMember.add(new Member(3, "m456", 12, "1990-12-01"));
System.out.println("Member当前顺序...");
System.out.println(listMember); // 方式一排序输出
System.out.println("Member默认排序(用自带的compareTo方法)后...");
Collections.sort(listMember);
System.out.println(listMember);
System.out.println("Member倒序(用自带的compareTo方法)后...");
Collections.sort(listMember, Collections.reverseOrder());
System.out.println(listMember); // 方式二排序输出
MySortList<Member> msList = new MySortList<Member>();
msList.sortByMethod(listMember, "getUsername", false);
System.out.println("Member按字段用户名排序后...");
System.out.println(listMember); msList.sortByMethod(listMember, "getLevel", false);
System.out.println("Member按字段级别排序后...");
System.out.println(listMember); msList.sortByMethod(listMember, "getBirthday", true);
System.out.println("Member按字段出生日期倒序后...");
System.out.println(listMember);
}
}

输出:

Member当前顺序...
[1|wm123|3|1992-12-01, 2|a234|8|1995-12-01, 3|m456|12|1990-12-01]
Member默认排序(用自带的compareTo方法)后...
[2|a234|8|1995-12-01, 3|m456|12|1990-12-01, 1|wm123|3|1992-12-01]
Member倒序(用自带的compareTo方法)后...
[1|wm123|3|1992-12-01, 3|m456|12|1990-12-01, 2|a234|8|1995-12-01]
Member按字段用户名排序后...
[2|a234|8|1995-12-01, 3|m456|12|1990-12-01, 1|wm123|3|1992-12-01]
Member按字段级别排序后...
[1|wm123|3|1992-12-01, 2|a234|8|1995-12-01, 3|m456|12|1990-12-01]
Member按字段出生日期倒序后...
[2|a234|8|1995-12-01, 1|wm123|3|1992-12-01, 3|m456|12|1990-12-01]

Java:集合,对列表(List)中的自定义对象按属性(字段)排序(正序、倒序)的方法的更多相关文章

  1. Java 集合 散列表hash table

    Java 集合 散列表hash table @author ixenos 摘要:hash table用链表数组实现.解决散列表的冲突:开放地址法 和 链地址法(冲突链表方式) hash table 是 ...

  2. 在配置文件(.settings、.config)中存储自定义对象

    原文:在配置文件(.settings..config)中存储自定义对象 引言 我前面曾写过一篇<使用配置文件(.settings..config)存储应用程序配置>,我在其中指出“sett ...

  3. JavaScript -- 时光流逝(四):js中的 Math 对象的属性和方法

    JavaScript -- 知识点回顾篇(四):js中的 Math 对象的属性和方法 1. Math 对象的属性 (1) E :返回算术常量 e,即自然对数的底数(约等于2.718). (2) LN2 ...

  4. array排序(按数组中对象的属性进行排序)

    使用array.sort()对数组中对象的属性进行排序 <template> <div> <a @click="sortArray()">降序& ...

  5. easyUI中datagrid展示对象下属性以及显示多个子属性(Day_37)

    easyUI中datagrid展示对象下属性以及显示多个子属性 显示对象单个属性值 添加formatter属性 <th field="decidedzone" width=& ...

  6. 使用 JavaScript 中的 document 对象的属性,根据下拉框中选择的属性,更改页面中的字体颜色和背景颜色

    查看本章节 查看作业目录 需求说明: 使用 JavaScript 中的 document 对象的属性,根据下拉框中选择的属性,更改页面中的字体颜色和背景颜色 实现思路: 在页面的 <body&g ...

  7. Python 关于拷贝(copy)汇总(列表拷贝 // 字典拷贝 // 自定义对象拷贝)

    1.列表拷贝 引用是指保存的值为对象的地址.在 Python 语言中,一个变量保存的值除了基本类型保存的是值外,其它都是引用,因此对于它们的使用就需要小心一些.下面举个例子: 问题描述:已知一个列表, ...

  8. JavaScript中创建自定义对象的方法

    本文内容参考JavaScript高级程序设计(第3版)第6章:面向对象的程序设计 ECMA-262中把对象定义为:“无序属性的集合,其属性可以包含基本值.对象或者函数.”我所理解的就是对象就是一个结构 ...

  9. Javascript 中创建自定义对象的方法(设计模式)

    Javascript 中创建对象,可以有很多种方法. Object构造函数/对象字面量: 抛开设计模式不谈,使用最基本的方法,就是先调用Object构造函数创建一个对象,然后给对象添加属性. var ...

随机推荐

  1. hdu 3308 LCIS(线段树区间合并)

    题目链接:http://acm.hdu.edu.cn/showproblem.php? pid=3308 LCIS Time Limit: 6000/2000 MS (Java/Others)     ...

  2. Flash:彻底理解crossdomain.xml、跨swf调用。

    安全域.crossdomain.xml,到处都有各种各种零碎的基础解释,所以这里不再复述这些概念. 本文目的是整理一下各种跨域加载的情况.什么时候会加载crossdomain,什么时候不加载.   1 ...

  3. python 怎么模拟加header(如User-Agent、Content-Type等等)

    # -*- coding: cp936 -*- #python 27 #xiaodeng #python 怎么模拟加header(如User-Agent.Content-Type等等) #办法一: i ...

  4. code vs 1013 求先序排列

    2001年NOIP全国联赛普及组 题目描述 Description 给出一棵二叉树的中序与后序排列.求出它的先序排列.(约定树结点用不同的大写字母表示,长度<=8). 输入描述 Input De ...

  5. SElinux测试及排错

    一.修改SElinux的状态 #sestatus --查看状态 #setenforce --临时修改 #setenforce #getenforce #vim /etc/selinux/config ...

  6. Dapper 的输出参数使用示范

    -- 普通SQL 示范-- Queries with output parameters. Hide Shrink Copy Code // output parameters // the para ...

  7. KMS11激活Window系列

    运行状态图 download: kms11

  8. 架构-虚拟路由器冗余协议【原理篇】VRRP详解

    转自:http://zhaoyuqiang.blog.51cto.com/6328846/1166840/ 为什么要使用VRRP技术 我们知道,为了实现不同子网之间的设备通信,需要配置路由.目前常用的 ...

  9. Spring使用内存数据库

    有时候为了做些测试需要创建数据库及相关表,安装MySQL等轻量数据库虽然简单但还是有点麻烦?而且用于自己临时测试的数据库对象一般不会被多次使用,还是浪费?内存数据库结合ORM可以很好解决这个问题. H ...

  10. (转)失败和拒绝,也是一种肯定 找工作时,我四处碰壁这一段经历对自己职业生涯的帮助最大。为什么? "因为这些挫折让我的脸皮变厚了 如果你不是每天被人拒绝,那就说明你的人生目标不够远大 所谓成功,就是不停地经历失败,并且始终保持热情

    (转)失败和拒绝,也是一种肯定 昨天,先是看到一个老外,说了一句很震撼的话. "你个人的项目,应该有四分之一会失败,否则就说明你的冒险精神不够." (Expect and hope ...