1.Set 存储的数据特点:无序的、不可重复的元素
具体的:以HashSet为例说明:
  1. 无序性:不等于随机性。存储的数据在底层数组中并非照数组索引的顺序添加,而是根据数据的哈希值决定的。
  2. 不可重复性:保证添加的元素照equals()判断时,不能返回true.即:相同的元素只能添加一个

2. 元素添加过程:(以HashSet为例)---HashSet底层:数组+链表的结构。(前提:jdk7)
我们向HashSet中添加元素a,首先调用元素a所在类的hashCode()方法,计算元素a的哈希值,此哈希值接着通过某种算法计算出在HashSet底层数组中的存放位置(即为:索引位置,判断数组此位置上是否已经元素:
    如果此位置上没其他元素,则元素a添加成功。 --->情况1
    如果此位置上其他元素b(或以链表形式存在的多个元素,则比较元素a与元素b的hash值:
        如果hash值不相同,则元素a添加成功。--->情况2
        如果hash值相同,进而需要调用元素a所在类的equals()方法:
               equals()返回true,元素a添加失败
               equals()返回false,则元素a添加成功。--->情况3
对于添加成功的情况2和情况3而言:元素a 与已经存在指定索引位置上数据以链表的方式存储。
jdk 7 :元素a放到数组中,指向原来的元素。
jdk 8 :原来的元素在数组中,指向元素a
总结:七上八下
 
3. 常用方法
Set接口中没额外定义新的方法,使用的都是Collection中声明过的方法。
 
4. 常用实现类:
|----Collection接口:单列集合,用来存储一个一个的对象
*          |----Set接口:存储无序的、不可重复的数据   -->高中讲的“集合”
*              |----HashSet:作为Set接口的主要实现类;线程不安全的;可以存储null值
*                  |----LinkedHashSet:作为HashSet的子类;遍历其内部数据时,可以按照添加的顺序遍历
*                 在添加数据的同时,每个数据还维护了两个引用,记录此数据前一个数据和后一个数据。                  
       对于频繁的遍历操作,LinkedHashSet效率高于HashSet.
*              |----TreeSet:可以照添加对象的指定属性,进行排序。
 
5. 存储对象所在类的要求:
  HashSet/LinkedHashSet:
要求:向Set(主要指:HashSet、LinkedHashSet)中添加的数据,其所在的类一定要重写hashCode() 和 equals()
为什么一定要重写hashCode() 和 equals():不全面的理解为默认情况下,hashCode返回的就是对象的存储地址,这样即使相同的对象也会因为不同的内存地址而被加入到Set中,但这违背Set的定义(不可重复)。因此一定要重写hashCode。此外,哈希值相同不一定是相同对象,因为存在碰撞冲突,因此还需要通过equals判断是否对象属性一致。但是不同的哈希值一定对应不同的对象。
要求:重写的hashCode()和equals()尽可能保持一致性:相等的对象必须具有相等的散列码
*    重写两个方法的小技巧:对象中用作 equals() 方法比较的 Field,都应该用来计算 hashCode 值。
*
  TreeSet:
要求:如果试图把一个对象添加到 TreeSet 时,则该对象的类必须实现 Comparable 接口。
1.自然排序中,比较两个对象是否相同的标准为:compareTo()返回0.
2.定制排序中,比较两个对象是否相同的标准为:compare()返回0. 需要将实现Comparator接口的实例作为形参传递给TreeSet的构造器。
  向 TreeSet 中添加元素时,只有第一个元素无须比较compareTo()方法,后面添 加的所有元素都会调用compareTo()方法进行比较。
  因为只有相同类的两个实例才会比较大小,所以向 TreeSet 中添加的应该是同 一个类的对象。
  对于 TreeSet 集合而言,它判断两个对象是否相等的唯一标准是:两个对象通 过 compareTo(Object obj) 方法比较返回值。
  当需要把一个对象放入 TreeSet 中,重写该对象对应的 equals() 方法时,应保 证该方法与 compareTo(Object obj) 方法有一致的结果:如果两个对象通过 equals() 方法比较返回 true,则通过 compareTo(Object obj) 方法比较应返回 0。 否则,让人难以理解。

Java比较器的使用背景:
Java中的对象,正常情况下,只能进行比较:==  或  != 。不能使用 > 或 < 的。但是在开发场景中,我们需要对多个对象进行排序,言外之意,就需要比较对象的大小。
  如何实现?使用两个接口中的任何一个:Comparable 或 Comparator
2.自然排序:使用Comparable接口
 2.1 说明
  1.像String、包装类等实现了Comparable接口,重写了compareTo(obj)方法,给出了比较两个对象大小的方式。
  2.像String、包装类重写compareTo()方法以后,进行了从小到大的排列
 3. 重写compareTo(obj)的规则:
     如果当前对象this大于形参对象obj,则返回正整数,
     如果当前对象this小于形参对象obj,则返回负整数,
     如果当前对象this等于形参对象obj,则返回零。
 4. 对于自定义类来说,如果需要排序,我们可以让自定义类实现Comparable接口,重写compareTo(obj)方法。在compareTo(obj)方法中指明如何排序
2.2 自定义类代码举例:
public class Goods implements Comparable{ private String name;
private double price; //指明商品比较大小的方式:照价格从低到高排序,再照产品名称从高到低排序
@Override
public int compareTo(Object o) {
// System.out.println("**************");
if(o instanceof Goods){
Goods goods = (Goods)o;
//方式一:
if(this.price > goods.price){
return 1;
}else if(this.price < goods.price){
return -1;
}else{
// return 0;
return -this.name.compareTo(goods.name);
}
//方式二:
// return Double.compare(this.price,goods.price);
}
// return 0;
throw new RuntimeException("传入的数据类型不一致!");
}
// getter、setter、toString()、构造器:省略
}
3.定制排序:使用Comparator接口
  3.1 说明
   1.背景:当元素的类型没实现java.lang.Comparable接口而又不方便修改代码,或者实现了java.lang.Comparable接口的排序规则不适合当前的操作,那么可以考虑使用 Comparator 的对象来排序
   2.重写compare(Object o1,Object o2)方法,比较o1和o2的大小:
    如果方法返回正整数,则表示o1大于o2;
    如果返回0,表示相等;
    返回负整数,表示o1小于o2。
  3.2 代码举例:
Comparator com = new Comparator() {
//指明商品比较大小的方式:照产品名称从低到高排序,再照价格从高到低排序
@Override
public int compare(Object o1, Object o2) {
if(o1 instanceof Goods && o2 instanceof Goods){
Goods g1 = (Goods)o1;
Goods g2 = (Goods)o2;
if(g1.getName().equals(g2.getName())){
return -Double.compare(g1.getPrice(),g2.getPrice());
}else{
return g1.getName().compareTo(g2.getName());
}
}
throw new RuntimeException("输入的数据类型不一致");
}
}

使用:

  Arrays.sort(goods,com);
  Collections.sort(coll,com);
  new TreeSet(com);
4. 两种排序方式对比
*    Comparable接口的方式一旦一定,保证Comparable接口实现类的对象在任何位置都可以比较大小。
*    Comparator接口属于临时性的比较。

6. TreeSet的使用
  6.1 使用说明:
    1.向TreeSet中添加的数据,要求是相同类的对象
    2.两种排序方式:自然排序(实现Comparable接口和定制排序(Comparator)
  6.2 常用的排序方式:
//方式一:自然排序
@Test
public void test1(){
TreeSet set = new TreeSet(); //失败:不能添加不同类的对象
// set.add(123);
// set.add(456);
// set.add("AA");
// set.add(new User("Tom",12)); //举例一:
// set.add(34);
// set.add(-34);
// set.add(43);
// set.add(11);
// set.add(8); //举例二:
set.add(new User("Tom",12));
set.add(new User("Jerry",32));
set.add(new User("Jim",2));
set.add(new User("Mike",65));
set.add(new User("Jack",33));
set.add(new User("Jack",56)); Iterator iterator = set.iterator();
while(iterator.hasNext()){
System.out.println(iterator.next());
} } //方式二:定制排序
@Test
public void test2(){
Comparator com = new Comparator() {
//照年龄从小到大排列
@Override
public int compare(Object o1, Object o2) {
if(o1 instanceof User && o2 instanceof User){
User u1 = (User)o1;
User u2 = (User)o2;
return Integer.compare(u1.getAge(),u2.getAge());
}else{
throw new RuntimeException("输入的数据类型不匹配");
}
}
}; TreeSet set = new TreeSet(com);
set.add(new User("Tom",12));
set.add(new User("Jerry",32));
set.add(new User("Jim",2));
set.add(new User("Mike",65));
set.add(new User("Mary",33));
set.add(new User("Jack",33));
set.add(new User("Jack",56)); Iterator iterator = set.iterator();
while(iterator.hasNext()){
System.out.println(iterator.next());
}
}
package TreeSetTest;

public class User implements Comparable {
private String name;
private int age; public User() {
} public User(String name, int age) {
this.name = name;
this.age = age;
} public String getName() {
return name;
} public int getAge() {
return age;
} public void setName(String name) {
this.name = name;
} public void setAge(int age) {
this.age = age;
} @Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", age=" + age +
'}';
} @Override
public int compareTo(Object o) {
if(o instanceof User){
User u=(User) o;
int compare=-this.name.compareTo(u.name); //String、包装类等实现了Comparable接口,重写了compareTo(obj)方法,进行了从小到大的排列
if(compare!=0){
return compare;
}else{
return -Integer.compare(u.age,this.age);//对于基本数据类型,采用包装类的compare方法
//return Integer.valueOf(u.age).compareTo(Integer.valueOf(this.age));
}
}
else {
throw new RuntimeException("输入类型不匹配");
} } @Override
public boolean equals(Object obj) {
System.out.println("equals");
if(this==obj)
return true;
if(obj==null || getClass()!=obj.getClass()) return false; User user=(User) obj;
// if(this.name.equals(u.name) && this.age==((User) obj).age)
// return true;
// else
// return false;
if (age != user.age) return false;
return name != null ? name.equals(user.name) : user.name == null;
} @Override
public int hashCode() {
int result= name!=null? name.hashCode():0;
result=31*result+age;
return result;
}
}

参考

 
关于底层实现:

 

最新文章

  1. 《连载 | 物联网框架ServerSuperIO教程》- 6.并发通讯模式开发及注意事项
  2. dd——留言板再加验证码功能
  3. 2.第一个Struts2程序-HelloWorld程序
  4. IOS-KVO&amp;KVC
  5. php 文件读取
  6. Keypress – 超强大!捕获键盘输入的 JavaScript 库
  7. 《疯狂Java:突破程序员基本功的16课》读书笔记-第一章 数组与内存控制
  8. JavaScript的apply和call方法及其区别
  9. WebService引擎Axis2完美教程
  10. justAP1.3.0版发布了
  11. filereader api 类型
  12. Python Selenium 之生成Beautiful可视化报告
  13. H-ui框架信息图标点击跳出页面问题
  14. 问题:Linux报swap空间占用过高,但物理内存还有空余
  15. wireshark抓本地回环包
  16. BZOJ1202 [HNOI2005]狡猾的商人 spfa
  17. ql.io来自ebay的api快速集成的构建api的框架
  18. 将excel数据分块多线程导入
  19. NO.4:自学python之路------内置方法、装饰器、迭代器
  20. K. Random Numbers(Gym 101466K + 线段树 + dfs序 + 快速幂 + 唯一分解)

热门文章

  1. python学习Day21
  2. ClickHouse 对付单表上亿条记录分组查询秒出, OLAP应用秒杀其他数据库
  3. 专门为小白准备的入门级mybatis-plus-generator代码自动生成器,提高开发效率。值得收藏
  4. SQL注入的几种类型
  5. 6┃音视频直播系统之 WebRTC 核心驱动SDP规范协商
  6. 忽略https域名校验不通过
  7. 429. N-ary Tree Level Order Traversal - LeetCode
  8. 好客租房53-context的使用
  9. [持续更新] Python学习、使用过程中遇见的非代码层面知识(想不到更好的标题了 T_T)
  10. PyTorch框架起步