首页 > 其他分享 >Set 接口及实现类

Set 接口及实现类

时间:2023-07-06 17:12:58浏览次数:44  
标签:Set hash 实现 接口 链表 key null

Set 接口及实现类

Set 接口基本介绍

  1. 无序(添加和取出顺序不一致),没有索引

  2. 不允许重复,所以最多包含一个null

  3. JDK API 中Set 接口实现类

image

Set 接口和常用方法

常用方法

和 List 接口一样,Set 接口也是 Collection 的子接口,因此,常用方法和 Collection 接口一样

Set 接口的遍历方式

同 Collection 接口的遍历方式一样,因为 Set 接口是 Collection 接口的子接口

  1. 使用迭代器
  2. 增强for
  3. 不能使用索引的方式来遍历

Set 接口的实现类 —— HashSet

  1. HashSet 实现了 Set 接口

    image

  2. HashSet 实际上是 HashMap

    public HashSet() {
        map = new HashMap<>();
    }
    
  3. 可以存放 null 值,但只能有一个

  4. HashSet 不保证元素是有序的,顺序取决于 hash 后,再确定索引的结果

  5. 不能有重复元素

HashSet 底层机制说明☆

  • HashSet 的底层是 HashMap,HashMap 的底层是 数组 + 链表 + 红黑树

  • Java中 HashMap 是利用“拉链法”处理 hashcode 的碰撞问题

image

  1. 先获取元素的哈希值hashCode()方法)

  2. 对哈希值进行运算,得出一个索引值即为要存放在哈希表中的位置号

  3. 找到存储数据表 table,如果该位置上没有其他元素,则直接存放

    如果该位置上已经有有其他元素,则要进行equals()方法判断,如果相等,则不再添加,如果不相等,则以链表的方式添加

  4. 在Java 8 中,如果一条链表的元素个数超过 TREEIFY_THRESHOLD(默认为8),并且 table的大小 >= MIN_TREEIFY_CAPACITY(默认为64)就会进行树化(红黑树

public class HashSetSource {
    public static void main(String[] args) {
        HashSet hashSet = new HashSet();
    }
}
  •  public HashSet() {
            map = new HashMap<>();
    }
    
    • public HashMap() {
              this.loadFactor = DEFAULT_LOAD_FACTOR; // all other fields defaulted
      }
      
hashSet.add("java");
  • public boolean add(E e) {
            return map.put(e, PRESENT)==null;
        // Dummy value to associate with an Object in the backing Map 与后置映射中的对象关联的虚拟值 
        // private static final Object PRESENT = new Object();
    }
    
    • public V put(K key, V value) {
              return putVal(hash(key), key, value, false, true);
      }
      
        • static final int hash(Object key) {
                  int h;
                  return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
          }
          
          • public int hashCode() { // 调用了该对象(这里是String)的hashCode()方法
                    int h = hash;
                    if (h == 0 && value.length > 0) {
                        char val[] = value;
            
                        for (int i = 0; i < value.length; i++) {
                            h = 31 * h + val[i];
                        }
                        hash = h;
                    }
                    return h;
            }
            
      • final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
                           boolean evict) {
                Node<K,V>[] tab; Node<K,V> p; int n, i; 
                if ((tab = table) == null || (n = tab.length) == 0) // table 是 HashMap 的数组,类型是 Node[]
                    // table 为null 或者 大小=0,第一次扩容到16
                    n = (tab = resize()).length;
        	// 根据key,得到hash,去计算该key应该存放在 table 的哪个索引位置
        	// 并把这个位置的对象,赋给p
            // 判断p是否为null,表示还没有存放元素,就创建一个 Node(key = "java", value=PRESENT)
                if ((p = tab[i = (n - 1) & hash]) == null)
                   
                    tab[i] = newNode(hash, key, value, null);
                else {
                    Node<K,V> e; K k; // 开发技巧,在需要局部变量(辅助变量)的时候,再创建
                    if (p.hash == hash &&
                        ((k = p.key) == key || (key != null && key.equals(k))))
                        // 如果当前索引位置对应的链表第一个元素和准备添加的key的hash值一样,并且
                        //(1)准备加入的key 和 p 指向的Node 结点的 key 是同一个对象
                        // (2) p指向的Node 结点的 key 的euqals() 和准备加入的 可以比较后相同
                        e = p;
                    else if (p instanceof TreeNode) // 判断p是不是一棵红黑树
                        // 如果是一颗红黑树,就调用putTreeVal()进行添加
                        e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
                    else { // 如果table对应的索引位置,已经是一个链表,就使用for循环比较
                        // (1)一次和该链表的每一个元素比较后,都不相同,则加入到该链表的对吼
                        for (int binCount = 0; ; ++binCount) {
                            if ((e = p.next) == null) {
                                p.next = newNode(hash, key, value, null);
                                if (binCount >= TREEIFY_THRESHOLD - 1) // binCount >= 8 - 1 链表长度等于8
                                    treeifyBin(tab, hash); // 树化
                                // 该函数里面进行判断是 table 扩容还是树化
                                // if (tab == null || (n = tab.length) < MIN_TREEIFY_CAPACITY) // <64 resize() // 扩容
                                // else if 树化
                                break;
                            }
                            if (e.hash == hash &&
                                ((k = e.key) == key || (key != null && key.equals(k))))
                                break;
                            p = e;
                        }
                    }
                    if (e != null) { // existing mapping for key
                        V oldValue = e.value;
                        if (!onlyIfAbsent || oldValue == null)
                            e.value = value;
                        afterNodeAccess(e);
                        return oldValue;
                    }
                }
                ++modCount;
                if (++size > threshold) // 当前为 16 * 0.75 = 12
                    resize(); // 扩容
                afterNodeInsertion(evict); // 空方法,HashMap留给子类去实现的方法
                return null;
            }
        

扩容和转成红黑树的机制☆☆

  1. HashSet 底层是HashMap,第一次添加时,table 数组扩容到 16,临界值(threshold)= 16 * 加载因子(LoadFactor =0.75)= 12

  2. 如果 table 数组到达临界值 12,就扩容到 16 * 2 = 32,新的临界值就是 32 * 0.75 = 24,以此类推

  3. 在Java 8 中,如果一条链表的元素个数到达 TREEIFY_THRESHOLD(默认为8),并且 table 的大小 >= MIN_TREEIFY_CAPACITY(默认为64),就会进行树化(红黑树),否则仍然采用数组扩容机制

Set 接口实现类 —— LinkedHashSet

LinkedHashSet 的说明

  1. LinkedHashSet 是 HashSet 的子类

image

  1. LinkedHashSet 底层是 LinkedHashMap,底层维护了一个 数组 + 双向链表

    数组 table 存放的结点类型是 LinkedHashMap$Entry

    static class Entry<K,V> extends HashMap.Node<K,V> {
            Entry<K,V> before, after;
            Entry(int hash, K key, V value, Node<K,V> next) {
                super(hash, key, value, next);
            }
    }
    
  2. LinkedHashSet 根据元素的 hashCode 值来确定元素的存储位置,同时使用链表维护元素的次序,这使得元素看起来是以插入顺序保存的

  3. LinkedHashSet 不允许重复元素

    image

    image

public static void main(String[] args) {
        LinkedHashSet linkedHashSet = new LinkedHashSet();

        linkedHashSet.add(new String("AA"));
        linkedHashSet.add(456);
        linkedHashSet.add(456);
        linkedHashSet.add(new Customer("刘", 1001));
        linkedHashSet.add(123);
        linkedHashSet.add(new Customer("张", 1002));
    // ...
}

image

  • LinkedHashSet 底层是LinkedHashMap,而LinkedHashMap 在 HashMap 存储结构的基础上多使用了双向链表记录添加元素的顺序
  • LinkedHashMap 中定义了 Entry 继承了HashMap的Node,底层存储数据时是数组+链表/红黑树(即哈希表结构),在此结构之上还定义了 Entry<K,V> before, after;用来记录添加元素的顺序

Set 接口实现类 —— TreeSet

TreeSet 的特点:

  1. 有序

    TreeSet可以确保集合元素处于排序状态。TreeSet支持两种排序方式,自然排序 和定制排序,其中自然排序为默认的排序方式。向TreeSet中加入的应该是同一个类的对象。

  2. 不重复;key不能为null(总结一点:凡是有Tree的集合,都是有序的,凡是有Set的就是不重复的)

  3. TreeSet 底层调用 TreeMap

  4. 自然排序使用要排序元素的 CompareTo(Object obj)方法来比较元素之间大小关系,然后将元素按照升序排列

    定制排序,应该使用 Comparator 接口,实现 int compare(T o1, T o2)方法。

标签:Set,hash,实现,接口,链表,key,null
From: https://www.cnblogs.com/ai135/p/17532708.html

相关文章

  • API接口的重要性
    API接口的重要性在现代软件开发中无可替代。以下是API接口的几个重要方面:1.实现系统集成:API接口允许不同应用程序之间实现数据共享和交流。通过API接口,不同的软件系统可以相互连接和协作,实现系统集成。这样可以提高系统的功能和效率,让不同系统之间实现无缝衔接。2.增加开发效......
  • 想了解API接口,这一篇就够了
    API(ApplicationProgrammingInterface)接口,对于大多数人来说可能还比较陌生,但实际上我们每天都在与它打交道。无论是使用手机上的应用程序,还是在网上购物,都少不了API接口的应用。那么,到底什么是API接口呢?如何调用API接口来获取淘宝商品数据呢?本文将为大家详细解答。什么是API接口......
  • 题解-Codeforces Round 805 (Div. 3) E. Split Into Two Sets
    题解-CodeforcesRound805(Div.3)E.SplitIntoTwoSets(原题链接)[Problem-E-Codeforces]思路知识点:种类并查集网上关于种类并查集的教学已经很多,在此不赘述在理解种类并查集时,很多文章会提到“敌人”,“朋友”的概念。而在不同的题目中,互为“敌人”,“朋友”的两个......
  • java实现文本转语音(即语音朗读)
    java实现文本转语音(即语音朗读)1.方式一:使用jacob离线语音合成1.下载jacob-1.18.zip链接:https://pan.baidu.com/s/1-zYB9I4VF5cPuj3ok1WLyg提取码:7t1g2.将jacob-1.18-x64.dll拷贝到jdk的bin目录或windows/SysWOW64目录中3.添加需要的依赖<!--https://mvnrepository.com/a......
  • Django restframwork中使用分页及实现自定义分页
    关于为何要分页以及如何在Django+Template架构中如何使用分页,可以参考之前的文章django自定义分页类和使用总结[1]DjangoRestFramework中分页限制今天开篇我们先不讲如何使用,我们先说Django+restframework实现前后端分离项目开发时,分页功能使用的限制?缘由是之前在开发运维......
  • API接口技术开发心得,阿里巴巴中国站获得1688商品详情数据采集商品规格信息列表调用参
     1688商品详情API接口的重要性主要体现在以下几个方面:提供全面的商品信息:1688商品详情API接口可以提供详尽的商品信息,包括商品名称、规格、价格、产地、供应商信息等。这些信息对于用户来说是非常重要的,可以帮助用户全面了解商品的特点和属性,从而做出更明智的购买决策。......
  • ios系统微信浏览器打开H5,调用接口status = 0失败的问题?
    最近写了一个很简单的小项目,以为不会有什么问题,今天突然说出问题了,说ios用户打开没有请求到数据。经测试,安卓,pc,都没有问题,只有ios出问题了。因为这次的涉及到时间,我以为ios时间处理上出问题了,仔细看了看,并不是,于是开启了漫长的寻找bug的过程。使用vConsole查看接口请求情况,发现......
  • Dynamics CRM字段安全配置文件,实现某个人只能看某条记录的某个字段
    共享安全字段https://blog.csdn.net/bzpfly/article/details/115652147 具体代码写法:https://learn.microsoft.com/zh-cn/power-apps/developer/data-platform/webapi/reference/fieldpermission?view=dataverse-latest  ......
  • C/C++ 实现VA与FOA之间的转换
    PE结构中的地址互转,这次再来系统的复习一下关于PE结构中各种地址的转换方式,最终通过编程来实现自动解析计算,最后将这个功能集成到我的迷你解析器中,本章中使用的工具是上次讲解PE结构文章中制作的CMD迷你结构解析器,如果不知道参数的基本使用请看前一篇。PE工具的使用与下载:C/C++实......
  • IBM总线代理接口SoupAction does not match
    IBM总线代理接口SoupActiondoesnotmatch问题描述:ThegivenSOAPActionuploadScheduledoesnotmatchanoperation.解决方案:增加一个ESQL:CREATECOMPUTEMODULETEST_Compute1CREATEFUNCTIONMain()RETURNSBOOLEANBEGINCALLCopyMessageHeaders();CALLCopy......