HashSet集合的add()方法的源码解析

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
简介: 一般来说,不同的字符串的哈希值是不同的。 1 package cn.itcast_02; 2 3 /* 4 * 一般来说,不同的字符串的哈希值是不同的。 5 * 哈希值仅仅是逻辑值,可能一样。

一般来说,不同的字符串的哈希值是不同的。

 1 package cn.itcast_02;
 2 
 3 /*
 4  *        一般来说,不同的字符串的哈希值是不同的。 
 5  *            哈希值仅仅是逻辑值,可能一样。
 6  *            地址值是实际的物理值,不一样。
 7  */
 8 
 9 public class HashCodeDemo {
10     public static void main(String[] args) {
11         System.out.println("hello".hashCode()); // 99162322
12         System.out.println("hello".hashCode()); // 99162322
13         System.out.println("world".hashCode()); // 113318802
14     }
15 }

 HashSet存储字符串并遍历

 1 package cn.itcast_02;
 2 
 3 import java.util.HashSet;
 4 
 5 /*
 6  * HashSet:存储字符串并遍历
 7  * 
 8  * 问题:为什么存储字符串的时候,字符串内容相同的只存储了一个呢?
 9  *         通过查看add方法的底层源码,我们知道这个方法的底层依赖两个方法:hashCode()和equals()。
10  *             即:
11  *                int hashCode()
12  *                boolean equals(Object obj)
13  * 
14  * 步骤:
15  *         首先比较哈希值
16  *             如果相同,继续走,比较地址值或者走equals()
17  *             如果不同,就直接添加到集合中    
18  * 
19  * 按照方法的步骤来说:    
20  *         先看hashCode()值是否相同
21  *             相同:继续走equals()方法
22  *                 返回true:说明元素重复,就不添加
23  *                 返回false:说明元素不重复,就添加到集合
24  *             不同:就直接把元素添加到集合中
25  * 
26  * 如果类没有重写这两个方法,默认使用的Object的方法。一般来说不相同。
27  * 而String类重写了hashCode()和equals()方法,所以,它就可以把内容相同的字符串去掉。只留下一个元素。
28  */
29 public class HashSetDemo {
30     public static void main(String[] args) {
31         // 创建集合对象
32         HashSet<String> hs = new HashSet<String>();
33 
34         // 创建并添加元素
35         hs.add("hello");
36         hs.add("world");
37         hs.add("java");
38         hs.add("world");
39 
40         // 遍历集合
41         for (String s : hs) {
42             System.out.println(s);
43         }
44     }
45 }

HashSet集合的add()方法的源码

---------------------------------------
interface Collection {
    ...
}

interface Set extends Collection {
    ...
}
---------------------------------------
class HashSet implements Set {
    ...
    private static final Object PRESENT = new Object();
    private transient HashMap<E,Object> map;
    
    public HashSet() {
        map = new HashMap<>();
    }
    
    public boolean add(E e) { // e = hello,world
        return map.put(e, PRESENT)==null;
    }
    ...
}
---------------------------------------
class HashMap implements Map {
  ...
public V put(K key, V value) { // key = e = hello,world // 看哈希表是否为空,如果是空,就开辟空间,开辟完空间,程序就往下走 if (table == EMPTY_TABLE) { inflateTable(threshold); } // 判断对象是否为null,不是null,程序就往下走 if (key == null) return putForNullKey(value); int hash = hash(key); // 和对象的hashCode()方法相关,即和对象的哈希值相关 int i = indexFor(hash, table.length); // 在哈希表中查找hash值 for (Entry<K,V> e = table[i]; e != null; e = e.next) { // 注意:这次的e其实是第一次/第二次的hello/world Object k; if (e.hash == hash && ((k = e.key) == key || key.equals(k))) { V oldValue = e.value; e.value = value; e.recordAccess(this); return oldValue; // 走到这里说明没有添加元素 } } modCount++; addEntry(hash, key, value, i); // 把元素添加进集合 return null; } ... transient int hashSeed = 0; ... final int hash(Object k) { // k = key = e = hello, int h = hashSeed; // transient int hashSeed = 0; if (0 != h && k instanceof String) { return sun.misc.Hashing.stringHash32((String) k); } h ^= k.hashCode(); // 这里调用的是对象的hashCode()方法 // This function ensures that hashCodes that differ only by // constant multiples at each bit position have a bounded // number of collisions (approximately 8 at default load factor). h ^= (h >>> 20) ^ (h >>> 12); return h ^ (h >>> 7) ^ (h >>> 4); } ... } --------------------------------------- hs.add("hello"); hs.add("world"); hs.add("java"); hs.add("world"); ---------------------------------------

 

我的GitHub地址: https://github.com/heizemingjun
我的博客园地址: http://www.cnblogs.com/chenmingjun
我的蚂蚁笔记博客地址: http://blog.leanote.com/chenmingjun
Copyright ©2018 黑泽明军
【转载文章务必保留出处和署名,谢谢!】
相关文章
|
3天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
14 2
|
3天前
|
存储 安全 Linux
Golang的GMP调度模型与源码解析
【11月更文挑战第11天】GMP 调度模型是 Go 语言运行时系统的核心部分,用于高效管理和调度大量协程(goroutine)。它通过少量的操作系统线程(M)和逻辑处理器(P)来调度大量的轻量级协程(G),从而实现高性能的并发处理。GMP 模型通过本地队列和全局队列来减少锁竞争,提高调度效率。在 Go 源码中,`runtime.h` 文件定义了关键数据结构,`schedule()` 和 `findrunnable()` 函数实现了核心调度逻辑。通过深入研究 GMP 模型,可以更好地理解 Go 语言的并发机制。
|
16天前
|
消息中间件 缓存 安全
Future与FutureTask源码解析,接口阻塞问题及解决方案
【11月更文挑战第5天】在Java开发中,多线程编程是提高系统并发性能和资源利用率的重要手段。然而,多线程编程也带来了诸如线程安全、死锁、接口阻塞等一系列复杂问题。本文将深度剖析多线程优化技巧、Future与FutureTask的源码、接口阻塞问题及解决方案,并通过具体业务场景和Java代码示例进行实战演示。
36 3
|
10天前
|
存储 Java 开发者
Java中的集合框架深入解析
【10月更文挑战第32天】本文旨在为读者揭开Java集合框架的神秘面纱,通过深入浅出的方式介绍其内部结构与运作机制。我们将从集合框架的设计哲学出发,探讨其如何影响我们的编程实践,并配以代码示例,展示如何在真实场景中应用这些知识。无论你是Java新手还是资深开发者,这篇文章都将为你提供新的视角和实用技巧。
11 0
|
1月前
|
缓存 Java 程序员
Map - LinkedHashSet&Map源码解析
Map - LinkedHashSet&Map源码解析
66 0
|
1月前
|
算法 Java 容器
Map - HashSet & HashMap 源码解析
Map - HashSet & HashMap 源码解析
52 0
|
1月前
|
存储 Java C++
Collection-PriorityQueue源码解析
Collection-PriorityQueue源码解析
60 0
|
1月前
|
安全 Java 程序员
Collection-Stack&Queue源码解析
Collection-Stack&Queue源码解析
80 0
|
1月前
|
存储
让星星⭐月亮告诉你,HashMap的put方法源码解析及其中两种会触发扩容的场景(足够详尽,有问题欢迎指正~)
`HashMap`的`put`方法通过调用`putVal`实现,主要涉及两个场景下的扩容操作:1. 初始化时,链表数组的初始容量设为16,阈值设为12;2. 当存储的元素个数超过阈值时,链表数组的容量和阈值均翻倍。`putVal`方法处理键值对的插入,包括链表和红黑树的转换,确保高效的数据存取。
53 5
|
1月前
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
107 5

推荐镜像

更多