Java自然排序 Comparable的使用

Posted 我永远信仰

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java自然排序 Comparable的使用相关的知识,希望对你有一定的参考价值。

抛出异常:java.lang.ClassCastException: com.collection.Student cannot be cast to java.lang.Comparable

寻找解决问题的方法:

去jdk文档里看看Comparable接口


原因是我们实体类拥有多个属性,将其存进TreeSet中不知道根据那个属性进行自然排序。实体类类需要实现Comparable这个接口,并重写他的自然比较方法compareTo,来定义排序规则

这里举个例子说明:

自然排序 Comparable的使用

题目要求:

存储学生对象并遍历,创建TreeSet集合使用无参构造方法

按照年龄从小到大排序,年龄相同时,按照姓名的字母顺序排序

//学生类
public class Student {
    private String name;
    private int age;

    public Student() {
    }

    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

}

//测试类:主方法
public class DemoTreeSet {
    public static void main(String[] args) {
        //无参构造
        TreeSet<Student> students = new TreeSet<>();
        //添加元素
        students.add(new Student("貂蝉", 23));
        students.add(new Student("王昭君", 29));
        students.add(new Student("西施", 18));
        students.add(new Student("杨玉环", 20));
        //直接运行,程序异常 Comparable比较器异常
        //java.lang.ClassCastException: com.collection.Student cannot be cast to java.lang.Comparable
        
        //遍历集合
        for (Student student : students) {
            System.out.println(student.getName()+" "+student.getAge());
        }
    }
}

原因是我们没有让学生类自然排序,所以学生类需要实现这个接口,并重写他的自然比较方法

public class Student implements Comparable<Student>{ //实现接口
    private String name;
    private int age;

    public Student() {
    }

    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }

    //比较方法
    @Override
    public int compareTo(Student o) {
        return 0;//注意这里返回值是0
    }
    
    ...
    ...
}

**运行程序:**我们添加的是四个学生,发现只输出了一个貂蝉。

原因在于我们在学生类的比较器中的返回值是0,集合中无元素时,第一个元素能加进来。而后的元素因为需要比较,而返回值是0,它会认为比较的这两个元素是同一个,所以添加失败。

那么我们将返回值改为1.

发现所有的元素都添加进来了。它是按照我们存储的顺序输出来的。

因为返回值是1,他会认为后面的元素会比前面的元素要大!所以应该添加在后面。


同理,如果改为-1,输出的是添加顺序的倒序。

那我们的要求是按照年龄从小到大排序,年龄相同时,按照姓名的字母顺序排序

先看按年龄排序

@Override
public int compareTo(Student o) {
    return this.age-o.age;//this是当前的,o是前面的,如果是个正数,当前的会加在后面。
}

输出的是


没问题,都存储进来了,而且年龄是按照升序排序。

但是如果又多了一个学生

students.add(new Student("西门庆", 29));

他的年龄和王昭君的一样,那他是否能添加成功呢。

显然是不行的。因为我们只对年龄进行了比较。他们两个年龄一样,会认为这两个是相同的元素,添加失败。

完整的比较

@Override
public int compareTo(Student o) {
    int num1 = this.age - o.age;
    int num2 = this.name.compareTo(o.name);//compareTo方法大于返回1,相等返回0,小于返回-1
    return num1 == 0 ? num2:num1;
}

运行结果

总结:

  • 用TreeSet集合存储自定义对象,无参构造方法使用的是自然排序对元素进行排序的
  • 自然排序,就是让元素所属的类实现Comparable接口,重写compareTo(T o)方法
  • 重写方法时,一定要注意排序规则必须按照要求的主要条件和次要条件来写

以上是关于Java自然排序 Comparable的使用的主要内容,如果未能解决你的问题,请参考以下文章

java的comparable接口啥意思

java lang(Comparable接口) 和java util(Comparator接口)分析比较

Comparable接口与Comparator接口的比较————Comparable接口详解

Comparable

自然排序Comparable的使用

Day029 Java比较器