HashMap插入相同key问题

目录
  • HashMap插入相同key
    • HashMap插入的描述
    • 我的问题
    • 想法
  • HashMap的key能不能重复
    • 我们看看实际代码
    • 说下重点

HashMap插入相同key

HashMap插入的描述

使用HashMap在插入操作时,会通过equal方法判断key是否相同。如果相同,则将覆盖对应的value;不相同才使用新的“桶”。

我的问题

当往HashMap中插入数据,即使有相同的key,但是能不能不进行覆盖操作,而是把新的value放在原有的value附近能够找到的位置?

想法

呃,其实大概方向就是通过一个HashMap<Integer, ArrayList>实现。。。

贴上代码

import java.util.ArrayList;
import java.util.HashMap; 
public class MapAndLink { 
    public static void main(String[] args){ 
        HashMap<Integer, ArrayList> map = new HashMap<>();
        put(1, 1, map);
        put(1, 3, map);
        put(2, 2, map);
        put(3, 4, map);
        put(1, 3, map);
        System.out.println(map.toString());
    }
 
    public static void put(Integer key, Integer str, HashMap<Integer, ArrayList> map){
        ArrayList<Integer> list = map.get(key);
        if(list == null)
            list = new ArrayList();
        for(int i = 0; i < list.size(); ++i){
            if(list.get(i).equals(str))
                return;
        }
        list.add(str);
        map.put(key, list);
    } 
}

再贴上输出结果

{1=[1, 3], 2=[2], 3=[4]}

HashMap的key能不能重复

今天测试同学给我提了一个bug,我看了bug描述很快定位了bug问题原因,乍一看这个问题的时候我当时很惊讶,HashMap的key应该是不允许重复的啊,为啥我put了两个一样的key,map中居然存入了两个一模一样个的key值,真是奇了怪了

我们看看实际代码

public enum Week {
    WED;
}
public class T {
    public static void main(String[] args) {

        Map a = new HashMap();
        // 枚举类型的一个 WED
        a.put(Week.WED, "星期二");
        a.put("WED", "星期三");
        System.out.println(a);
    }
}

我们debug看一下这个map的返回结果:

看到结果没有,两个一模一样的key,看到这里我就稍稍有一点懵了,这时我马上想到了Hashmap的底层实现,其中put方法的底层是调用putVal(has(key),val,....),key是由传入的参数经过hascode()计算得出的,既然map里存了两个相同的key,那么它们的hascode一定是不相同的,继续看has()方法的底层,实际就是调用了传入这个Object.hascode(),这时其实答案就已经浮出水面了,问题点就是出在我传入的两个key的类型上,虽然我们表面看枚举WED和字符串的值是相同的,可是它们对应的类型一个是String,一个是Enum,这时候我们继续看源码可以发现,String和Enum的hascode是完全不同的

从这里我们可以看出来,实际我们存入map中的key是我们的枚举类Week,而另一个则是String类型的一个字符串,既然知道了问题的原因,那么我们就有了相应的解决办法。

我们知道在使用枚举类型时,如果不指定枚举中常量的值,默认是直接返回这个常量name的,但是返回的这个name并不是String类型,而是一个Enum类型,所以在使用枚举类型时候一定要注意。

说下重点

  • 在判断常量值是否相等的时候,要么重新value方法,要么使用name().toString()方法来进行校验。
  • 除此之外,我们在使用HashMap的时候,如果在不指定<?,?>key和value的类型的时候,切记不要想当然的认为看到的key相同,存入的key就一定是相同的。

结论:Hashmap中的key是不允许重复的,我们看到的重复只是程序欺骗了我们的双眼。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • 关于HashMap相同key累加value的问题

    目录 HashMap相同key累加value HashMap解决key值相同问题 HashMap相同key累加value import java.util.HashMap; import java.util.Map; public class Test { public static void main(String[] args) { Map<String, Long> map = new HashMap<String, Long>(); map.put("k"

  • 自定义对象作为HashMap的Key问题

    目录 自定义对象作为HashMap的Key 首先我们自定义一个学生对象,它有姓名和年龄两个字段. HashMap使用自定义对象作为Key的注意点 1. 自定义对象不重写hashCode方法和equals会发生什么? 2. 在HashMap中使用自定义对象作为key会发生什么? 3. 重写hashCode方法和equals方法的原则 自定义对象作为HashMap的Key 这个问题在很多面试者面试时都会被提及,本人也是最近在看effective java第九条:覆盖equals时总要覆盖hashco

  • HashMap实现保存两个key相同的数据

    HashMap如何保存两个key相同的数据 最近一个朋友去面试了,面试官问了一个关于HashMap的问题:HashMap如何保存两个key相同的数据? 准确来说,应该是Map中如何保存两个key相同的数据,因为用来实现这个功能的IdentityHashMap类和HashMap虽然都是实现了Map接口,但本质是属于不同的东西: 我们知道在HashMap中,如果key相同就会被覆盖,那IdentityHashMap是怎么实现这个功能的呢? java jdk源码中,IdentityHashMap类上写

  • 解决Java Redis删除HashMap中的key踩到的坑

    现象 Java使用Redis删除HashMap中的key时,取出对应的HashMap后通过Java中HashMap的remove方法移除key然后重新调用redis的Hmset方法将覆盖无效 示例代码 //通过key取出对应的HashMap Map<String, String> ruleMap = jedisCluster.hgetAll("HashKey"); //通过java中移除HashMap中的Key ruleMap.remove("ruleA"

  • HashMap插入相同key问题

    目录 HashMap插入相同key HashMap插入的描述 我的问题 想法 HashMap的key能不能重复 我们看看实际代码 说下重点 HashMap插入相同key HashMap插入的描述 使用HashMap在插入操作时,会通过equal方法判断key是否相同.如果相同,则将覆盖对应的value:不相同才使用新的“桶”. 我的问题 当往HashMap中插入数据,即使有相同的key,但是能不能不进行覆盖操作,而是把新的value放在原有的value附近能够找到的位置? 想法 呃,其实大概方向

  • 浅谈HashMap、HashTable的key和value是否可为null

    结论: HashMap对象的key.value值均可为null. HahTable对象的key.value值均不可为null. 且两者的的key值均不能重复,若添加key相同的键值对,后面的value会自动覆盖前面的value,但不会报错. public class Test { public static void main(String[] args) { Map<String, String> map = new HashMap<String, String>();//Has

  • Java中HashMap里面key为null存放到哪

    我们知道HashMap集合是允许存放null值的 hashMap是根据key的hashCode来寻找存放位置的,那当key为null时, 怎么存储呢? 在put方法里头,其实第一行就处理了key=null的情况. // HashMap的put方法 public V put(K key, V value) { if (table == EMPTY_TABLE) { inflateTable(threshold); } if (key == null) // key为null调用putForNull

  • Java中HashMap和TreeMap的区别深入理解

    首先介绍一下什么是Map.在数组中我们是通过数组下标来对其内容索引的,而在Map中我们通过对象来对对象进行索引,用来索引的对象叫做key,其对应的对象叫做value.这就是我们平时说的键值对. HashMap通过hashcode对其内容进行快速查找,而 TreeMap中所有的元素都保持着某种固定的顺序,如果你需要得到一个有序的结果你就应该使用TreeMap(HashMap中元素的排列顺序是不固定的). HashMap 非线程安全 TreeMap 非线程安全 线程安全 在Java里,线程安全一般体

  • 在Java8与Java7中HashMap源码实现的对比

    一.HashMap的原理介绍 此乃老生常谈,不作仔细解说. 一句话概括之:HashMap是一个散列表,它存储的内容是键值对(key-value)映射. 二.Java 7 中HashMap的源码分析 首先是HashMap的构造函数代码块1中,根据初始化的Capacity与loadFactor(加载因子)初始化HashMap. //代码块1 public HashMap(int initialCapacity, float loadFactor) { if (initialCapacity < 0)

  • JAVA HashMap详细介绍和示例

    第1部分 HashMap介绍HashMap简介HashMap 是一个散列表,它存储的内容是键值对(key-value)映射.HashMap 继承于AbstractMap,实现了Map.Cloneable.java.io.Serializable接口.HashMap 的实现不是同步的,这意味着它不是线程安全的.它的key.value都可以为null.此外,HashMap中的映射不是有序的.HashMap 的实例有两个参数影响其性能:"初始容量" 和 "加载因子".容量

  • Java 中的HashMap详解和使用示例_动力节点Java学院整理

    第1部分 HashMap介绍 HashMap简介 HashMap 是一个散列表,它存储的内容是键值对(key-value)映射. HashMap 继承于AbstractMap,实现了Map.Cloneable.java.io.Serializable接口. HashMap 的实现不是同步的,这意味着它不是线程安全的.它的key.value都可以为null.此外,HashMap中的映射不是有序的. HashMap 的实例有两个参数影响其性能:"初始容量" 和 "加载因子&quo

随机推荐