ClassCastException: com.collection.Student cannot be cast to java.lang.Comparable

本文详细解析了Java中TreeSet集合利用自然排序进行元素排序的问题。当存储自定义对象时,由于未指定排序规则,会抛出ClassCastException。解决方法是让实体类实现Comparable接口并重写compareTo方法,根据需求定制排序规则。文中通过学生类实例展示了如何按照年龄和姓名进行排序,并分析了不同比较返回值对排序的影响,强调了正确实现排序规则的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

抛出异常: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)方法
  • 重写方法时,一定要注意排序规则必须按照要求的主要条件和次要条件来写
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值