【自然排序】

 package com.hxl;

 public class Student implements Comparable<Student> {

     private String name;
private int age; public Student() {
super();
} public Student(String name, int age) {
super();
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;
} @Override
public int compareTo(Student s) {
// 先让两个对象的age属性做差比较,这个是主要排序条件
int num = this.age - s.age;
// 若age属性相同,再比较name属性(String类本身实现了Comparable接口)
// 即在主要排序条件相同的情况下,次要排序条件起作用
int flag = num == 0 ? this.name.compareTo(s.name) : num;
// 返回比较结果
return flag;
}
}
 package com.hxl;

 import java.util.TreeSet;

 public class Test {
public static void main(String[] args) {
//这里使用的无参构造实例化TreeSet集合,则默认启用的是自然排序
TreeSet<Student> ts = new TreeSet<Student>();
ts.add(new Student("cc", 11));
ts.add(new Student("ee", 11));
ts.add(new Student("cc", 22));
ts.add(new Student("aa", 22));
ts.add(new Student("bb", 11)); for (Student s : ts) {
System.out.println(s.getName()+"_"+s.getAge());
} /*
为什么TreeSet集合中的元素既唯一又有序呢?
原因是它在存储元素的时候就是有序存储的(红黑树结构存储)
TreeSet的add()方法底层依赖的是Comparable的compareTo方法
这里就是说元素类本身要有自己的compareTo方法
所以元素类本身必须实现Comparable接口,重写compareTo方法
compareTo方法有个特点:它返回的是int型数据,结果有三类负数、0、正数
例如:(Java中一些常见的有比较意义的一些类都实现了Comparable接口,如Integer类)
Integer a = new Integer(10);
Integer b = new Integer(20);
int num = a.compareTo(b); //因为a小于b,所以num返回的是负数
而TreeSet的add()方法这样理解此返回值:
即返回负数则比根节点小,元素在此集合中唯一,元素存放根的左孩子
返回正数则比根节点大,元素在此集合中唯一,元素存放根的右孩子
返回0则表示,元素相同,在此集合中不唯一,故而丢掉不存放
由此可见,我们的重写的compareTo()方法决定了TreeSet集合中元素的去留和顺序!
*/
}
}

【比较器排序(外部类实现)】

 package com.hxl;

 public class Student{

     private String name;
private int age; public Student() {
super();
} public Student(String name, int age) {
super();
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;
}
}
 package com.hxl;

 import java.util.Comparator;

 public class MyComparator implements Comparator<Student> {

     @Override
public int compare(Student s1, Student s2) {
// 先让两个对象的age属性做差比较,这个是主要排序条件
int num = s1.getAge() - s2.getAge();
// 若age属性相同,再比较name属性(String类本身实现了Comparable接口)
// 即在主要排序条件相同的情况下,次要排序条件起作用
int flag = num == 0 ? s1.getName().compareTo(s2.getName()) : num;
// 返回比较结果
return flag;
}
}
 package com.hxl;

 import java.util.TreeSet;

 public class Test {
public static void main(String[] args) {
//这里使用TreeSet(Comparator comparator)构造实例化TreeSet集合,则启用的是指定比较器排序
TreeSet<Student> ts = new TreeSet<Student>(new MyComparator());
ts.add(new Student("cc", 11));
ts.add(new Student("ee", 11));
ts.add(new Student("cc", 22));
ts.add(new Student("aa", 22));
ts.add(new Student("bb", 11)); for (Student s : ts) {
System.out.println(s.getName()+"_"+s.getAge());
}
}
}

【比较器排序(内部类实现,如果只使用一次的话)】

 package com.hxl;

 public class Student{

     private String name;
private int age; public Student() {
super();
} public Student(String name, int age) {
super();
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;
}
}
 package com.hxl;

 import java.util.Comparator;
import java.util.TreeSet; public class Test {
public static void main(String[] args) {
//如果一个方法的参数是接口,那么真实想要的是其实是接口实现类的对象
//这里的对象只用一次,专门定义一个外部类显得麻烦
//匿名内部类可以实现这个需求
TreeSet<Student> ts = new TreeSet<Student>(new Comparator<Student>(){
public int compare(Student s1, Student s2) {
// 先让两个对象的age属性做差比较,这个是主要排序条件
int num = s1.getAge() - s2.getAge();
// 若age属性相同,再比较name属性(String类本身实现了Comparable接口)
// 即在主要排序条件相同的情况下,次要排序条件起作用
int flag = num == 0 ? s1.getName().compareTo(s2.getName()) : num;
// 返回比较结果
return flag;
}
});
ts.add(new Student("cc", 11));
ts.add(new Student("ee", 11));
ts.add(new Student("cc", 22));
ts.add(new Student("aa", 22));
ts.add(new Student("bb", 11)); for (Student s : ts) {
System.out.println(s.getName()+"_"+s.getAge());
}
}
}

【注】开发中会用最后一种,因为第一种只有固定的排序方式,第二种每次都要定义外面类显得麻烦。

TreeSet集合的自然排序与比较器排序、Comparable接口的compareTo()方法的更多相关文章

  1. 什么是泛型?,Set集合,TreeSet集合自然排序和比较器排序,数据结构-二叉树,数据结构-平衡二叉树

    ==知识点== 1.泛型 2.Set集合 3.TreeSet 4.数据结构-二叉树 5.数据结构-平衡二叉树 ==用到的单词== 1.element[ˈelɪmənt] 要素 元素(软) 2.key[ ...

  2. TreeSet对非自然顺序元素的排序

    /* 1. 往TreeSet添加元素的时候,如果元素本身具备了自然顺序的特性,那么就按照元素自然顺序的特性进行排序存储. 2. 往TreeSet添加元素的时候,如果元素本身不具备自然顺序的特性,那么该 ...

  3. TreeSet集合如何保证元素唯一

    TreeSet: 1.特点 TreeSet是用来排序的, 可以指定一个顺序, 对象存入之后会按照指定的顺序排列 2.使用方式 a.自然顺序(Comparable) TreeSet类的add()方法中会 ...

  4. java通过Comparable接口实现字符串比较大小排序的简单实例

    /** * 对象比较大小compare的用法 字符串排序 * 练习代码, 给定字符串" nba" "cba" "ncaa" "wb ...

  5. Java—集合框架 Collections.sort()、Comparable接口和Comparator接口

    Collentions工具类--java.util.Collections Collentions是Java集合框架中,用来操作集合对象的工具类,也是Java集合框架的成员,与List.Map和Set ...

  6. 《java入门第一季》之集合框架TreeSet存储元素自然排序以及图解

    这一篇对TreeSet做介绍,先看一个简单的例子: * TreeSet:能够对元素按照某种规则进行排序.  * 排序有两种方式  * A:自然排序: 从小到大排序  * B:比较器排序    Comp ...

  7. TreeSet集合的add()方法源码解析(01.Integer自然排序)

    >TreeSet集合使用实例 >TreeSet集合的红黑树 存储与取出(图) >TreeSet的add()方法源码     TreeSet集合使用实例 package cn.itca ...

  8. TreeSet集合排序方式一:自然排序Comparable

    TreeSet集合默认会进行排序.因此必须有排序,如果没有就会报类型转换异常. 自然排序 Person class->实现Comparable,实现compareTo()方法 package H ...

  9. TreeSet 比较器排序 自定义对象

    package cn.itcast.day21.treeset2; import java.util.Comparator; import java.util.TreeSet; /* * TreeSe ...

随机推荐

  1. Java中instanceof与getClass的区别

    在比较一个类和另一个类是否属于同一个类实例的时候,通常可以采用instanceof和getClass两种方法比较两者是否相等来判断,但是两者在判断上面是有差别的,下面通过代码说明: public cl ...

  2. 前端 --- 关于DOM的介绍

    111 什么是DOM DOM:文档对象模型.DOM 为文档提供了结构化表示,并定义了如何通过脚本来访问文档结构.目的其实就是为了能让js操作html元素而制定的一个规范. DOM就是由节点组成的. 解 ...

  3. mysql定时任务,每天的零点执行一个存储过程

    1 前言 利用navicat工具来写存储过程及定时执行,此文章是按照自身经验总结的,仅作为记录使用. 2 步骤 2.1 新建过程 2.2 在函数体写你需要执行的代码 CREATE DEFINER=`r ...

  4. 41)django-admin

    一:介绍 通过django admin可以快速生成后台管理功能. 二:设置 工程同名下settings.py 1)在INSTALLED_APPS中增加django.contrib.admin 2)在I ...

  5. 16)django-ajax使用

    通过ajax可以悄悄的把数据传输给服务器,实现页面无刷新. 一:ajax使用语法 1)普通方式 ajax使用语法: $.ajax({ url:"/host", //提交到那里 ty ...

  6. Codeforces 1132G Greedy Subsequences [线段树]

    洛谷 Codeforces 看到题解那么少就来发一篇吧-- 思路 看完题目一脸懵逼,感觉无从下手. 莫名其妙地想到笛卡尔树,但笛卡尔树好像并没有太大作用. 考虑把笛卡尔树改一下:每个点的父亲设为它的右 ...

  7. liunx 安装 mysql 5.6

    第一步  解压文件 目录切换到/usr/local/ cd /usr/local/ 解压 tar zxvf mysql-5.6.33-linux-glibc2.5-x86_64.tar.gz 重命名为 ...

  8. 一道关于js声明变量,var和let的面试题

    function aa(flag) { // var test // 变量提升,函数最顶部 if(flag) { var test = 'hello man' } else { //此处访问 test ...

  9. leetcode(js)算法之914卡牌分组

    给定一副牌,每张牌上都写着一个整数. 此时,你需要选定一个数字 X,使我们可以将整副牌按下述规则分成 1 组或更多组: 每组都有 X 张牌. 组内所有的牌上都写着相同的整数. 仅当你可选的 X > ...

  10. mabatis的批量新增sql 初级的 初级的 初级的

    简单描述:做开发的时候,会遇到一次插入好多条记录,怎么做好呢? 解决思路:循环insert啊!  哪凉快那呆着去←!←  这样会增加数据库开销的,当然不能这么干了,要在sql上下功夫.看代码,一下就明 ...