java treeset原理_跟我学java编程—Set接口实现类TreeSet

前面两节分别介绍了Set接口的实现类HashSet和LinkedHashSet类。本节介绍TreeSet类。

TreeSet 也是一个有序的集合,TreeSet使用二叉树的原理对加入的元素按照升序或降序排序,该类对新添加的元素都会进行排序,将元素插入到二叉树指定的位置。

下面先通过一个应用实例对TreeSet的用法有个具体了解。

5b23685f1d55778b2de11e89a585ad84.png

与HashSet集合采用通过hash算法来决定元素的存储位置不同,TreeSet采用二叉树的数据结构来存储集合元素。

527f166cca36bf8eca748eb7a196b13e.png

图 14-6 TreeSetTest示例输出结果

从输出结果可以看出,TreeSet对加入元素默认按照升序进行排序。

TreeSet的排序原理

TreeSet调用元素的compareTo方法来比较元素之间的大小关系,然后将元素按升序或降序排列,因此待加入的元素必须实现compareTo方法,并通过compareTo方法比较两个元素内容的大小。

待加入的元素如何实现compareTo方法呢?Java提供了一个Comparable接口,该接口定义了一个compareTo(Object o)方法,用于比较两个元素的大小,并返回一个整数值。需要加入TreeSet集合的元素必须要实现该接口,该接口的实现代码用于比较当前元素与传入元素的大小进行比较,并返回结果。因此实现该接口类的对象就可以比较大小。当一个对象调用该方法与另一个对象进行比较时,该方法就可以返回比较结果。例如:tempObj1.compareTo(obj2),如果该方法返回0,则代表这两个对象相等;如果该方法返回1,则表明tempObj1大于obj2;如果该方法返回-1,则表明tempObj1小于obj2。

需要加入TreeSet集合的元素必须实现Comparable接口,如果试图把没有实现Comparable接口的类对象加入TreeSet时,程序就会抛出异常。

没有实现Comparable接口类的程序代码如下:

8e7cfccb6daec50d7e5929946aba1f65.png

MyNumber为自定义的序号类,该类没有实现Comparable接口。程序在MainTest类的main方法中声明了TreeSet集合,并添加MyNumber对象。运行上述代码后,程序抛出错误,如下图所示:

a0e5fb393ec5096ae84e9e64a05ad60f.png图 14-7 MainTest程序输出结果

从输出结果可以看出,程序抛出“Exception in thread "main" java.lang.ClassCastException: treesetnum.MyNumber cannot be cast to java.lang.Comparable”异常,意思是MyNumber类没有实现Comparable接口,导致进行元素比较时发生错误。

在上面的例子代码中,TreeSet集合添加了三个MyNumber对象,添加第一个对象时,TreeSet里没有任何元素,所以不会出现任何问题,当添加第二个MyNumber对象时,TreeSet就会调用该对象的compareTo(Object obj)方法与集合中的其他元素进行比较,如果其对应的类没有实现Comparable 接口,则会引发ClassCastException异常。

修改上面的代码如下:

405a58a5db48b65fdee3ebcaff1bb887.png

修改后的代码在MyNumber类中实现了Comparable接口,实现方法名为compareTo,该方法实现两个类属性num的大小比较。如果当前num大于传入的num,该方法返回1;如果当前num小于传入的num,该方法返回-1;若相等则返回0。

修改后的代码输出结果如下图所示:

e473384258489b36b7e1561f97cd3fd8.png图 14-8 修改后代码输出结果

从图中可以看出,修改后的代码可以正常地加入元素,并能对加入的元素进行正确排序。

■ 知识点拨

在实际编程中,需要注意的是放入TreeSet集合中的元素必须是可“排序”的,对加入的元素(自定义类),若要实现compareTo方法,必须实现Comparable接口。在Java SDK中,已经有一些实现了Comparable接口的类,其中包括String字符串类,还包括Float、Interger、Long、Short、Double等数字类。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值