Java高级编程--常用类之比较器

在Java中经常会涉及到对象数组的排序问题,那么就涉及到对象之间的比较问题。Java中的对象,正常情况下,只能进行比较:==或!=,不能使用>或<,但是在实际开发中,有时我们需要对多个对象进行排序,也就是比较对象的大小,此时需要使用Comparable接口或Comparator接口。


Java实现对象排序的方式有两种:

➢ 自然排序:java.lang.Comparable
➢ 定制排序:java.util.Comparator

▾ 自然排序:java.lang.Comparable

自然排序要点:

➢ Comparable接口强行对实现它的每个类的对象进行整体排序。这种排序被称为类的自然排序。

➢ 实现 Comparable的类必须实现 compareTo(Object obj) 方法,两个对象即通过compareTo(Object obj)方法的返回值来比较大小。通常为如果当前对象this大于形参对象obj,则返回正整数,如果当前对象this小于形参对象obj,则返回负整数,如果当前对象this等于形参对象obj,则返回零。

➢ 实现Comparable接口的对象列表(和数组)可以通过 Collections.sort 或Arrays.sort进行自动排序。实现Comparable接口的对象可以用作有序集合中的元素,如果要使用集合中关于比较的方法,通常要先实现对象的Comparable接口。

➢ 最好使类的自然排序和equals一致。即: e1.compareTo(e2) == 0 与e1.equals(e2) 具有相同的 boolean 值。虽然非必须要求,单使用时建议最好保持equals一致。

Comparable的典型使用:

ღ 以下实现默认都是从小到大排序

✣ String:按照字符串中字符的Unicode值进行比较

✣ Character(char包装类):按照字符的Unicode值来进行比较

✣ 数值类型对应的包装类以及BigInteger、BigDecimal:按照它们对应的数值大小进行比较

✣ Date、Time等:靠后的日期时间比靠前的日期时间大

使用例子:

➀ 重写CompareTo()的规则:如果当前对象this大于形参对象obj,则返回正整数,如果当前对象this小于形参对象obj,则返回负整数,如果当前对象this等于形参对象obj,则返回零。

➁ 对于自定义类来说,如果需要排序,我们可以让自定义类实现Comparable接口并重写CompareTo(obj)方法在CompareTo(obj方法中定义如何排序) ---- 自然排序。

//商品类实现Comparable接口
public class CompareGoods implements Comparable{
    private String name;
    private double price;
    public CompareGoods(String name, double price){
        this.name = name;
        this.price = price;
    }

    public double getPrice() {return price;}
    public void setPrice(double price) { this.price = price; }
    public String getName() { return name;}
    public void setName(String name) { this.name = name;}

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        CompareGoods that = (CompareGoods) o;
        return Double.compare(that.price, price) == 0 &&
                Objects.equals(name, that.name);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, price);
    }

    @Override
    public String toString() {
        return "CompareGoods{" +
                "name='" + name + '\'' +
                ", price=" + price +
                '}';
    }

    /**
     * @Description: 按照价格从低到高比较商品,价格相同按照商品名排序
     */
    @Override
    public int compareTo(Object o) {
        if (o instanceof CompareGoods){
            //方式一
            CompareGoods goods = (CompareGoods)o;
            if (this.price < goods.price){
                return -1;
            }else if(this.price > goods.price){
                return 1;
            }else {
                return this.name.compareTo(goods.name);
            }
            //方式二
            //return Double.compare(this.price,goods.price);
        }
        throw new RuntimeException("传入的数据类型不一致");
    }
}

实现

//测试Comparable接口实现类
class CompareTest {
    @Test
    public void testComparable(){
        String[] arr = new String[]{"cc","aB","EGa","吃货","ge*&","gF35"};
        Arrays.sort(arr);
        System.out.println(Arrays.toString(arr));
        System.out.println("ef".compareTo("Age"));  //正数,ef比Age大
    }

    /*
        自定义类实现Comparable自然排序
     */
    @Test
    public void myObjectComparable(){
        CompareGoods goods[] = new CompareGoods[5];
        goods[0] = new CompareGoods("lenovoMouse",58);
        goods[1] = new CompareGoods("haweiMouse",68);
        goods[2] = new CompareGoods("microsoft",51);
        goods[3] = new CompareGoods("miMouse",38);
        goods[4] = new CompareGoods("dellMouse",51);
        Arrays.sort(goods);
        //ClassCastException未实现比较器的自定义类不能使用sort方法,
        // 需要实现Comparable接口后才能使用Arrays.sort(arr[])方法
        System.out.println(Arrays.toString(goods));
        System.out.println(goods[0].compareTo(goods[1]));  //-1
    }
}

▾ 定制排序:java.util.Comparator

➢ 当元素的类型没有实现java.lang.Comparable 接口而又不方便修改代码,或者实现了java.lang.Comparable 接口的排序规则不适合当前的操作,那用么可以考虑使用Comparator的对象来排序,强行对多个对象进行整体排序的比较。

➢ 重写compare(Object o1,Object o2)方法,比较o1和o2的大小:如果方法返回正整数,则表示o1大于o2 ;如果返回0 ,表示相等;返回负整数,表示o1小于o2。

➢ 可以将 Comparator传递给 sort 方法(如 Collections.sort或 Arrays.sort),从而允许在排序顺序上实现精确控制。

➢ 还可以使用 Comparator来控制某些数据结构(如有序 set或有序映射)的顺序,或者为那些没有自然顺序的对象collection提供排序。

使用例子:

//商品类实现Comparator接口,比较器
public class GoodsComparator implements Comparator{
    private String name;
    private double price;
    public CompareGoods(String name, double price){
        this.name = name;
        this.price = price;
    }

    public double getPrice() {return price;}
    public void setPrice(double price) { this.price = price; }
    public String getName() { return name;}
    public void setName(String name) { this.name = name;}

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        CompareGoods that = (CompareGoods) o;
        return Double.compare(that.price, price) == 0 &&
                Objects.equals(name, that.name);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, price);
    }

    @Override
    public String toString() {
        return "CompareGoods{" +
                "name='" + name + '\'' +
                ", price=" + price +
                '}';
    }

    /**
     * @Description: 按照按照商品名从低到高排序,商品名相同价格从低到高比较商品
     */
    @Override
    public int compare(Object o1, Object o2) {
         if (o1 instanceof CompareGoods && o2 instanceof CompareGoods){
              CompareGoods goods1 = (CompareGoods)o1;
              CompareGoods goods2 = (CompareGoods)o2;
              if (goods1.getName().equals(goods2.getName())){
                   return Double.compare(goods1.getPrice(),goods2.getPrice());
               }else {
                   return goods1.getName().compareTo(goods2.getName());
               }
          }
          throw new RuntimeException("传入的数据类型不一致");
      }
}

实现

//测试Comparator接口实现类
class ComparatorTest {

    /*
        自定义类实现Comparable自然排序
     */
    @Test
    public void myObjectComparable(){
        CompareGoods goods[] = new CompareGoods[6];
        goods[0] = new CompareGoods("lenovoMouse",58);
        goods[1] = new CompareGoods("haweiMouse",68);
        goods[2] = new CompareGoods("microsoft",51);
        goods[3] = new CompareGoods("miMouse",38);
        goods[4] = new CompareGoods("dellMouse",51);
        goods[5] = new CompareGoods("haweiMouse",98);
        GoodsComparator comp = new GoodsComparator();
        Arrays.sort(goods,comp);
        System.out.println(comp.compare(goods[0], goods[1]));  //-4
    }
}

Compare和Comparator的异同别

➢ 两者都可以实现自定义类对象的比较,也都可以参与Arrays.sort()和Collections.sort()的排序方法。

➢  实现Compare接口的类,其对象可以直接调用compareTo方法来与其他对象比较,要求比较的对象类型一致或可自动转型。

➢  Comparator接口实现类相当于一个工具了,在比较时创建Comparator实现类对象,调用compare方法传入两个要比较的对象,类型不一定相同,具体需要根据重写的compare方法中的比较规则来定,如按Boy和Girl的年龄比较。


本博客与CSDN博客༺ཌ༈君☠纤༈ད༻同步发布

posted @ 2020-09-14 09:19  ༺ཌ༈君☠纤༈ད༻  阅读(301)  评论(0编辑  收藏  举报