TreeSet是实现Set接口的实现类。所以它存储的值是唯一的,同时也可以对存储的值进行排序,排序用的是二叉树原理。所以要理解这个类,必须先简单理解一下什么是二叉树。

  • 二叉树原理简述

假如有这么一个集合TreeSet<Integer>是[5,11,6,5,23,14]

用二叉树是怎么排序的呢?

二叉树遍历方法比较多,有兴趣自己百度看下吧。这里只需要知道元素是怎么插入到二叉树即可。小的存储在左边(负数),大的存储在右边(正数),相等不存储。

  • TreeSet的基本使用

public static void main(String[] args) {
TreeSet<Integer> ts = new TreeSet<>();
ts.add(2);
ts.add(1);
ts.add(3);
ts.add(2);
ts.add(3);
ts.add(1);
ts.add(2);
System.out.println(ts);
} // 输出结果:
[1,2,3]

可以知道,TreeSet集合不仅可以保证集合元素的唯一性,还可以排序。

如果TreeSet里面存储的是对象呢?会出现什么情况呢?

public static void main(String[] args) {
TreeSet<Students> ts = new TreeSet<>();
ts.add(new Students("张三",13));
ts.add(new Students("李四",14));
ts.add(new Students("王五",15));
System.out.println(ts);
} // 输出结果:
Exception in thread "main" java.lang.ClassCastException: com.lei.Students cannot be cast to java.lang.Comparable

报错了,因为集合里面的是对象,对象不能转换为比较可比较对象。

如果想根据年龄排序,打印出各个对象(toString方法),应该怎么做呢?

在API里面搜索一下Comparable,发现是个接口,那么我们就可以让Students类实现Comparable接口方法,这样Students对象就成为了可比较对象了。

Students类实现Comparable接口方法:

public class Students implements Comparable<Students> {
private String name;
private int age; ...... @Override
public int compareTo(Students o) {
return this.age-o.age;
}
}

为什么是this.age-o.age?this.age代表调用时的对象的age,返回的如果是正数(比o.age大),就存储在右边。返回的是如果是负数(比o.age小),就存储在左边。如果等于0,就不存储。

这就出问题了,如果两个人不同名字,同样年龄,this.age - o.age = 0,不就存不进二叉树了吗?

验证一下:

public static void main(String[] args) {
TreeSet<Students> ts = new TreeSet<>();
ts.add(new Students("李四",14));
ts.add(new Students("张三",13));
ts.add(new Students("王五",15));
ts.add(new Students("赵六",13));
System.out.println(ts);
} // 输出结果只有张三、李四、王五

所以需要改进一下Students类的compareTo方法,保证同年龄,但是不同名字的学生也能存进二叉树。

@Override
public int compareTo(Students o) {
int num = this.age - o.age;
// String类里面已经重写了compareTo方法
// int compareTo(String anotherString) 按字典顺序比较两个字符串
return num == 0 ? this.name.compareTo(o.name) : num;

这样就可以把四个不同的对象存储进来,并且先按照年龄排序,年龄相同的再按照字符串排序。

除了这种方式可以实现排序以外,还有一种方式可以实现排序。

TreeSet有这么一个构造方法:

TreeSet(Comparator<? super E> comparator)      构造一个新的,空的树集,根据指定的比较器进行排序。

Comparator是什么呢?API文档看一下:

Interface Comparator<T>,是一个接口,里面有一个要实现的接口方法:

int    compare(T o1, T o2)   比较其两个参数的顺序。

例如,我们要对字符串的长度进行排序,长度相同的安装字符串排序:

public class Test5 {
public static void main(String[] args) {
TreeSet<String> ts = new TreeSet<>(new SortedByLen()); // 父类引用指向子类对象 Comparator c = new SortedByLen();
ts.add("aaaaaaaaa");
ts.add("wc");
ts.add("nba");
ts.add("cba");
ts.add("chichung");
System.out.println(ts);
}
} class SortedByLen implements Comparator<String> { @Override
public int compare(String o1, String o2) {
int num = o1.length() - o2.length();
return num == 0 ? o1.compareTo(o2) : num;
}
} //输出结果:
[wc, cba, nba, chichung, aaaaaaaaa]

需要注意的是重写compare方法的o1,o2。o1代表调用的对象,o2代表集合中的对象。

两种实现排序方式视情况而用。

(1)自然顺序(Comparable)

  • TreeSet类的add()方法中会把存入的对象提升为Comparable类型
  • 调用对象的compareTo()方法和集合中的对象比较
  • 根据compareTo()方法返回的结果进行存储

(2)比较器顺序(Comparator)

  • 创建TreeSet的时候可以指定一个Comparator
  • 如果传入了Comparator的子类对象,那么TreeSet就会按照比较器中的顺序排序
  • 调用的对象是compare方法的第一个参数,集合中的对象是compare方法的第二个参数

(3)两种方式的区别

  • TreeSet构造函数什么都不传,默认按照类中Comparable的顺序(没有就报错ClassCastException)
  • TreeSet如果传入Comparator,就优先按照Comparator

如果不想保证元素的唯一性,改一下compare方法就可以了,永远不要让它返回0。

详解java中的TreeSet集合的更多相关文章

  1. 详解Java中的clone方法

    详解Java中的clone方法 参考:http://blog.csdn.net/zhangjg_blog/article/details/18369201/ 所谓的复制对象,首先要分配一个和源对象同样 ...

  2. 【Java学习笔记之三十三】详解Java中try,catch,finally的用法及分析

    这一篇我们将会介绍java中try,catch,finally的用法 以下先给出try,catch用法: try { //需要被检测的异常代码 } catch(Exception e) { //异常处 ...

  3. 详解java中的数据结构

    线性表,链表,哈希表是常用的数据结构,在进行Java开发时,JDK已经为我们提供了一系列相应的类来实现基本的数据结构.这些类均在java.util包中.本文试图通过简单的描述,向读者阐述各个类的作用以 ...

  4. 详解Java中的Object.getClass()方法

    详解Object.getClass()方法,这个方法的返回值是Class类型,Class c = obj.getClass(); 通过对象c,我们可以获取该对象的所有成员方法,每个成员方法都是一个Me ...

  5. 详解Java中的字符串

    字符串常量池详解 在深入学习字符串类之前, 我们先搞懂JVM是怎样处理新生字符串的. 当你知道字符串的初始化细节后, 再去写String s = "hello"或String s ...

  6. 详解Java中的final关键字

    本文原文地址:https://jiang-hao.com/articles/2019/coding-java-final-keyword.html1 final 简介2 final关键字可用于多个场景 ...

  7. 详解Java中的clone方法:原型模式

    转:http://developer.51cto.com/art/201506/478985.htm clone顾名思义就是复制, 在Java语言中, clone方法被对象调用,所以会复制对象.所谓的 ...

  8. 详解Java中的访问控制修饰符(public, protected, default, private)

    Java中的访问控制修饰符已经困惑笔者多时,其中较复杂的情况一直不能理解透彻.今天下定决心,系统.全面地研究Java中的访问控制修饰符的所有方面,并整理成这篇文章,希望有同样疑惑的读者读完后能有所收获 ...

  9. java中String是对象还是类?详解java中的String

    有很多人搞不懂对象和类的定义.比如说java中String到底是对象还是类呢? 有人说String 既可以说是类,也可以说是对象. 其实他这么说也没问题, 类和对象其实都是一个抽象的概念. 我们可以把 ...

随机推荐

  1. POJ3261:Milk Patterns——题解

    http://poj.org/problem?id=3261 给一个序列,求至少出现 k 次的最长重复子串,这 k 个子串可以重叠. 论文题+傻逼题. 上一道题(POJ1743)会做即可. 还是二分长 ...

  2. POJ2135:Farm Tour——题解

    http://poj.org/problem?id=2135 题目大意: 从1到n再回来,每条边只能走一次,问最短路. —————————————————— 如果不告诉我是费用流打死不会想这个…… 我 ...

  3. HDU.1285 确定比赛名次 (拓扑排序 TopSort)

    HDU.1285 确定比赛名次 (拓扑排序 TopSort) 题意分析 裸的拓扑排序 详解请移步 算法学习 拓扑排序(TopSort) 只不过这道的额外要求是,输出字典序最小的那组解.那么解决方案就是 ...

  4. Change the IPTables log file

    http://www.networkinghowtos.com/howto/change-the-iptables-log-file/     An important aspect of any f ...

  5. Mybatis手工写sql语句及Mapper.xml方法

    首先在项目中 建一个mapper包,然后在spring集合mybatis的配置文件中设置扫描这个mapper包 然后,建 封装查询结果需要的 pojo 然后,在 mapper包中创建 Mapper接口 ...

  6. STL之三:deque用法详解

    转载于:http://blog.csdn.net/longshengguoji/article/details/8519812 deque函数: deque容器为一个给定类型的元素进行线性处理,像向量 ...

  7. Samba共享传输大文件(ex:1G)失败的问题

    1:问题描述 1.1 基本信息 遇见这样一个bug,路由器有USB share的功能,可将U盘内的文件通过samba和LAN端PC机中文件进行共享,测试发现小文件可正常共享,一旦文件大了(比如1G左右 ...

  8. SDUT3926 kmp

    bLue的二叉树 Time Limit: 3000MS Memory Limit: 65536KB Submit Statistic Problem Description Keke 是一个喜爱种树的 ...

  9. Foxmail安装和登录

    Foxmail安装和登录... ============================== 下载Foxmail客户端:http://www.foxmail.com/ ================ ...

  10. kafka命令总结

    ./kafka-console-consumer.sh  --bootstrap-server IP:9092  --topic user-asset-change-v1 --partition 2  ...