参考博客:【https://www.cnblogs.com/xrq730/p/4850140.html】
前言:
在开发中,有时候我们要用的排序,例如我们需要导出EXCEL表格,而根据SQL语句查询出来的数据排序不能满足我们的需求是,这时候就需要我们自定义排序,而这其中就需要用到Comparable和Comparator接口,在这里总结一下两者的区别:
一,Comparable接口:
Comparable可以认为是一个内比较器,实现了Comparable接口的类有一个特点,就是这些类是可以和自己比较的,至于具体和另一个实现了Comparable接口的类如何比较,则依赖compareTo方法的实现,compareTo方法也被称为自然比较方法。如果开发者add进入一个Collection的对象想要Collections的sort方法帮你自动进行排序的话,那么这个对象必须实现Comparable接口。compareTo方法的返回值是int,有三种情况:
1、比较者大于被比较者(也就是compareTo方法里面的对象),那么返回正整数
2、比较者等于被比较者,那么返回0
3、比较者小于被比较者,那么返回负整数
编写一个测试类让其实现Comparable接口且实现compareTo方法
public class User implements Comparable<User> { private Integer userID; private String userName; private String address; public User(Integer userID, String address, String userName){ this.userID = userID; this.address = address; this.userName = userName; } @Override public int compareTo(User o) { return this.userID.compareTo(o.userID); } public Integer getUserID() { return userID; } public void setUserID(Integer userID) { this.userID = userID; } public String getUserName() { return userName; } public void setUserName(String userName) { this.userName = userName; } public String getAddress() { return address; } public void setAddress(String address) { this.address = address; } }
public class Test { public static void main(String[] args) { List<User> list = new ArrayList<User>(); User user1 = new User(10,null,null); User user2 = new User(20,null,null); User user3 = new User(1,null,null); User user4 = new User(15,null,null); list.add(user1); list.add(user2); list.add(user3); list.add(user4); Collections.sort(list); for (User user: list){ System.out.println(user.getUserID()); } } }
测试结果:
Connected to the target VM, address: '127.0.0.1:54336', transport: 'socket' 1 10 15 20 Disconnected from the target VM, address: '127.0.0.1:54336', transport: 'socket' Process finished with exit code 0
二、Comparator接口:
Comparator可以认为是是一个外比较器,个人认为有两种情况可以使用实现Comparator接口的方式:
1、一个对象不支持自己和自己比较(没有实现Comparable接口),但是又想对两个对象进行比较
2、一个对象实现了Comparable接口,但是开发者认为compareTo方法中的比较方式并不是自己想要的那种比较方式
Comparator接口里面有一个compare方法,方法有两个参数T o1和T o2,是泛型的表示方式,分别表示待比较的两个对象,方法返回值和Comparable接口一样是int,有三种情况:
1、o1大于o2,返回正整数
2、o1等于o2,返回0
3、o1小于o3,返回负整数
修改一下上面的例子:
public class User { private Integer userID; private String userName; private String address; public User(Integer userID, String address, String userName){ this.userID = userID; this.address = address; this.userName = userName; } public Integer getUserID() { return userID; } public void setUserID(Integer userID) { this.userID = userID; } public String getUserName() { return userName; } public void setUserName(String userName) { this.userName = userName; } public String getAddress() { return address; } public void setAddress(String address) { this.address = address; } }
public class Test { public static void main(String[] args) { List<User> list = new ArrayList<User>(); User user1 = new User(10,null,null); User user2 = new User(20,null,null); User user3 = new User(1,null,null); User user4 = new User(15,null,null); list.add(user1); list.add(user2); list.add(user3); list.add(user4); Collections.sort(list, new Comparator<User>() { @Override public int compare(User o1, User o2) { return o1.getUserID().compareTo(o2.getUserID()); } }); for (User user: list){ System.out.println(user.getUserID()); } } }
测试结果:
1
10
15
20
Process finished with exit code 0
三、总结:
1、如果实现类没有实现Comparable接口,又想对两个类进行比较(或者实现类实现了Comparable接口,但是对compareTo方法内的比较算法不满意),那么可以实现Comparator接口,自定义一个比较器,写比较算法
2、实现Comparable接口的方式比实现Comparator接口的耦合性要强一些,如果要修改比较算法,要修改Comparable接口的实现类,而实现Comparator的类是在外部进行比较的,不需要对实现类有任何修改。