Java对象正常情况下,只能进行比较:== 或 !=,不能使用 > or < ,但是在开发中,可能需要对多个对象排序,言外之意,就需要比较对象的大小。
如何实现?使用两个接口中任意一个:Comparable 或 Comparator
Comparable接口----自然排序
- 像String、包装类等实现了Comparable接口,重写了compareTo(obj)方法,给出了比较两个对象大小的方式
- 像String、包装类等重写了compareTo方法以后,进行了从小到大的排列
- 重写compareTo(obj)的规则:
当前对象this大于形参对象bj,则返回正整数,
当前对象this小于形参对象obj,则返回负整数,
当前对象this等于形参对象obj,则返回零。
先给出下面会用到的商品类Goods,已实现Comparable接口
class Goods implements Comparable{
private String name;
private int price;
public Goods(String name, int price) {
this.name = name;
this.price = price;
}
public String getName() {
return name;
}
public int getPrice() {
return price;
}
public void setName(String name) {
this.name = name;
}
public void setPrice(int price) {
this.price = price;
}
@Override
public String toString() {
return "Goods{" +
"name='" + name + '\'' +
", price=" + price +
'}';
}
@Override
public int compareTo(Object o) {
if(o instanceof Goods){
Goods goods = (Goods) o;
// if(this.price> goods.price){
// return 1;
// }else if(this.price< goods.price){
// return -1;
// }else{
// return 0;
// }
return Integer.compare(this.price, goods.price);
}
throw new RuntimeException("传入的数据类型有误");
}
}
这里已经实现了Comparable接口,重写了compareTo()方法,Goods对象按照价格从低到高排列。
@Test
public void test1() {
String[] arr = new String[]{"AA", "CC", "KK", "MM", "QQ", "BB"};
Arrays.sort(arr);
System.out.println(Arrays.toString(arr));
//输出[AA,BB,CC,KK,MM,QQ]
}
@Test
public void test2() {
Goods[] arr = new Goods[4];
arr[0] = new Goods("iPhone", 9000);
arr[1] = new Goods("Mi 10pro", 3999);
arr[2] = new Goods("ONEPLUS 8pro", 5399);
arr[3] = new Goods("HUAWEI P40pro+", 7000);
Arrays.sort(arr);
System.out.println(Arrays.toString(arr));
/*输出
[Goods{name='Mi 10pro', price=3999},
Goods{name='ONEPLUS 8pro', price=5399},
Goods{name='HUAWEI P40pro+', price=7000},
Goods{name='iPhone', price=9000}]
*/
}
Comparator接口----定制排序
使用背景:
当元素的类型没有实现java. lang . Comparable接口而又不方便修改代码, 或者实现了java. Lang. Comparable接口的排序规则不适合当前的操作,那么可以考虑使用Comparator的对象来排序。
规则:
- 定义Comparator对象,写好内部比较逻辑,再传入方法中;
- 或者直接采用匿名对象的方式(下面代码片采用)。
重写compare(Object 01, object o2)方法,比较o1和o2的大小:
- 如果方法返回正整数,则表示o1大于o2; .
- 如果返回e,表示相等;
- 返回负整数,表示o1小于o2。
@Test
public void test3() {
String[] arr = new String[]{"AA", "CC", "KK", "M", "GG", "JJ", "DD"};
// Comparator(a,b)
// a - the array to be sorted
// b - the comparator to determine the order of the array. A null value
// indicates that the elements' natural ordering should be used
Arrays.sort(arr, new Comparator() {
//按照字符串从大到小的顺序排列
@Override
public int compare(Object o1, Object o2) {
if (o1 instanceof String && o2 instanceof String) {
String s1 = (String) o1;
String s2 = (String) o2;
return -s1.compareTo(s2);
}
throw new RuntimeException("输入的数据类型不一致");
}
});
System.out.println(Arrays.toString(arr));
}
@Test
public void test4() {
Goods[] arr = new Goods[5];
arr[0] = new Goods("iPhone", 9000);
arr[1] = new Goods("Mi 10pro", 3999);
arr[2] = new Goods("ONEPLUS 8pro", 5399);
arr[3] = new Goods("HUAWEI P40pro+", 7000);
arr[4] = new Goods("iPhone", 8000);
Arrays.sort(arr, new Comparator<Goods>() {
@Override
public int compare(Goods goods, Goods t1) {
if(goods.getName().equals(t1.getName())){
return Integer.compare(goods.getPrice(), t1.getPrice());
}else {
return goods.getName().compareTo(t1.getName());
}
}
});
System.out.println(Arrays.toString(arr));
}
两个接口的对比
Comparable接口与Comparator的使用的对比:
- Comparable接口的方式一旦确定,能保证Comparable接口实现类的对象在任何位置都可以比较大小
- Comparator接口属于临时性的比较,调用方法时再去制定规则。