Java中如何使用 byte 数组作为 Map 的 key

 更新时间:2023年06月07日 11:12:03   作者:万猫学社  
本文将讨论在使用HashMap时,当byte数组作为key时所遇到的问题及其解决方案,介绍使用String和List这两种数据结构作为临时解决方案的方法,感兴趣的朋友跟随小编一起看看吧

本文将引领我们探索:如何将byte数组作为HashMap中键。HashMap的机制使我们无法直接这样做。让我们研究一下,为何出现此状况,以及针对这种情况,几种可供选择的解决方案。

HashMap的工作原理

HashMap是一种使用哈希机制来存储和检索值的数据结构。使用哈希码来存储和检索值可以大大提高HashMap的性能,因为它可以使查找键值对的时间复杂度保持在 O ( 1 ) O(1) O(1)的级别。当然,这也要求我们在实现hashCode()方法时尽可能地让哈希码分布均匀,以免造成哈希冲突,从而影响HashMap的效率。

当我们调用put(key, value)方法时,HashMap会通过键的hashCode()方法计算哈希码。这个哈希码用于确定最终存储值的桶:

public V get(Object key) {
    Node<K,V> e;
    return (e = getNode(hash(key), key)) == null ? null : e.value;
}

在使用get(key)方法检索值时,需要经过一系列处理步骤:首先,会通过键计算哈希码,然后找到哈希桶。接下来,使用equals()方法检查桶中的每个条目是否与键相等。最终,返回匹配条目的值:

public V put(K key, V value) {
    return putVal(hash(key), key, value, false, true);
}

equalshashCode方法

在Java编程中,equals方法和hashCode方法都有应该遵守的规则。在HashMap这个数据结构中,有一个方面尤其重要:具有相同equals方法比较结果的对象,必须返回相同的哈希值。然而,反之则不一定成立,也就是说,具有相同哈希值的对象,并不一定具有相同的equals方法比较结果。这也是为什么我们可以将多个对象存储在HashMap的同一个桶中的原因。

在使用HashMap时,建议不要更改key的哈希值。虽然这不是强制性规定,但强烈建议将键定义为不可变对象。如果对象是不可变的,无论hashCode方法的实现如何,它的哈希值都不会被更改。

在默认情况下,哈希值是基于对象的所有字段进行计算的。如果我们需要使用可变的键,我们需要重写hashCode方法,以确保它的计算不涉及可变字段。为了维护这一个规则,我们还需要修改equals方法。

使用 byte 数组作为key

为了能够从映射中成功地检索值,相等性必须是有意义的。这就是使用byte数组并不是一个真正的选择的主要原因。在Java中,数组使用对象标识来确定相等性。如果我们使用byte数组作为key创建HashMap,那么只有使用完全相同的数组对象才能检索值。

让我们使用byte数组作为key创建一个简单的例子:

byte[] key1 = {1, 2, 3};
byte[] key2 = {1, 2, 3};
Map<byte[], String> map = new HashMap<>();
map.put(key1, "value1");
map.put(key2, "value2");
System.out.println(map.get(key1));
System.out.println(map.get(key2));
System.out.println(map.get(new byte[]{1, 2, 3}));

我们虽然有两个相同的键,但是我们无法使用具有相同值的新创建的数组检索到任何内容,运行结果如下:

value1
value2
null

解决方法

使用String

String的相等性基于字符数组的内容:

public boolean equals(Object anObject) {
    if (this == anObject) {
        return true;
    }
    if (anObject instanceof String) {
        String anotherString = (String)anObject;
        int n = value.length;
        if (n == anotherString.value.length) {
            char v1[] = value;
            char v2[] = anotherString.value;
            int i = 0;
            while (n-- != 0) {
                if (v1[i] != v2[i])
                    return false;
                i++;
            }
            return true;
        }
    }
    return false;
}

字符串也是不可变的,并且基于byte数组创建一个字符串非常简单。我们可以使用Base64轻松编码和解码字符串,然后创建一个使用字符串作为key而不是byte数组的HashMap:

String key1 = Base64.getEncoder().encodeToString(new byte[]{1, 2, 3});
String key2 = Base64.getEncoder().encodeToString(new byte[]{1, 2, 3});
Map<String, String> map = new HashMap<>();
map.put(key1, "value1");
map.put(key2, "value2");
System.out.println(map.get(key1));
System.out.println(map.get(key2));
System.out.println(map.get(Base64.getEncoder().encodeToString(new byte[]{1, 2, 3})));

运行结果如下:

value2
value2
value2

注意: 在byte数组转化为String时会有性能损耗。因此,在大多数情况下,该解决方案并不推荐

使用List

String类似,List#equals方法将检查其每个元素的相等性:

public boolean equals(Object o) {
    if (o == this)
        return true;
    if (!(o instanceof List))
        return false;
    ListIterator<E> e1 = listIterator();
    ListIterator<?> e2 = ((List<?>) o).listIterator();
    while (e1.hasNext() && e2.hasNext()) {
        E o1 = e1.next();
        Object o2 = e2.next();
        if (!(o1==null ? o2==null : o1.equals(o2)))
            return false;
    }
    return !(e1.hasNext() || e2.hasNext());
}

如果这些元素具有合理的equals()方法并且是不可变的,则List将作为HashMap键正确工作。我们只需要确保使用不可变的List实现:

List<Byte> key1 = ImmutableList.of((byte) 1, (byte) 2, (byte) 3);
List<Byte> key2 = ImmutableList.of((byte) 1, (byte) 2, (byte) 3);
Map<List<Byte>, String> map = new HashMap<>();
map.put(key1, "value1");
map.put(key2, "value2");
System.out.println(map.get(key1));
System.out.println(map.get(key2));
System.out.println(map.get(ImmutableList.of((byte) 1, (byte) 2, (byte) 3)));

运行结果如下:

value2
value2
value2

注意: Byte对象的列表将占用比byte数组更多的内存。因此,在大多数情况下,该解决方案并不推荐

自定义类(推荐使用)

我们还可以自己的定义一个类,用来完全控制哈希码计算和相等性。这样,我们可以确保解决方案快速,并且没有太大的内存占用。

让我们创建一个只有一个final私有byte数组字段的类。它将没有setter方法,只用getter方法,用来确保完全不可变性。

然后在实现自己的equalshashCode方法。为了方法,我们可以使用Arrays类来完成这两项任务,最终代码如下:

public class BytesKey {
    private final byte[] array;
    public BytesKey(byte[] array) {
        this.array = array;
    }
    public byte[] getArray() {
        return array.clone();
    }
    @Override
    public boolean equals(Object o) {
        if (this == o) {
            return true;
        }
        if (o == null || getClass() != o.getClass()){
            return false;
        }
        BytesKey bytesKey = (BytesKey) o;
        return Arrays.equals(array, bytesKey.array);
    }
    @Override
    public int hashCode() {
        return Arrays.hashCode(array);
    }
}

最后,我们使用我们自定义的类作为HashMap的key:

BytesKey key1 = new BytesKey(new byte[]{1, 2, 3});
BytesKey key2 = new BytesKey(new byte[]{1, 2, 3});
Map<BytesKey, String> map = new HashMap<>();
map.put(key1, "value1");
map.put(key2, "value2");
System.out.println(map.get(key1));
System.out.println(map.get(key2));
System.out.println(map.get(new BytesKey(new byte[]{1, 2, 3})));

运行结果如下:

value2
value2
value2

注意: 自定义的类既没有转化为String的性能损耗,也没有Byte对象列表的内存占用。因此,该解决方案推荐使用

总结

本文将讨论在使用HashMap时,当byte数组作为key时所遇到的问题及其解决方案。

首先,我们将研究为什么不能直接使用数组作为键。在使用HashMap时,我们需要保证每个键的唯一性,而使用数组作为键可能会出现冲突。这是因为数组的hashCode值是基于其在内存中的地址计算得出的,因此即使两个数组内容完全相同,它们在内存中的位置不同,它们的hashCode也会不同。因此,直接使用数组作为键可能会导致无法正确获取值或者出现意外的覆盖。

接着,我们会介绍使用String和List这两种数据结构作为临时解决方案的方法。它们都是具有可比性和可哈希性的数据结构,能够保证唯一性。但这种方法并不是完美的解决方案,因为使用String或List作为键会带来一些性能上的开销,或者占用不必要的内存空间。

最后,我们将通过自定义类的方式完美解决这个问题。这个自定义类包含了一个byte数组字段,并重写hashCodeequals方法,以确保唯一性和正确性。通过这种方式,我们可以避免使用String或List时的性能和内存占用问题,并且能够在保证正确性的同时获得更高的效率。

到此这篇关于在Java中使用 byte 数组作为 Map 的 key的文章就介绍到这了,更多相关Java使用 byte 数组作为 Map 的 key内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • java实现桌面右下角弹窗效果

    java实现桌面右下角弹窗效果

    这篇文章主要为大家详细介绍了java实现桌面右下角弹窗效果,模仿类似于qq消息弹窗,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-07-07
  • 初学者Android studio安装图文详解

    初学者Android studio安装图文详解

    本文给大家分享android studio 安装图文详解包括下载安装及遇到的问题,非常不错,具有参考借鉴价值,需要的朋友参考下吧
    2017-09-09
  • 布隆过滤器面试如何快速判断元素是否在集合里

    布隆过滤器面试如何快速判断元素是否在集合里

    这篇文章主要为大家介绍了布隆过滤器面试中如何快速判断元素是否在集合里的完美回复,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步
    2022-03-03
  • Java实现单人信息管理程序

    Java实现单人信息管理程序

    这篇文章主要为大家详细介绍了Java实现单人信息管理程序,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-02-02
  • springcloud之自定义简易消费服务组件

    springcloud之自定义简易消费服务组件

    这篇文章主要介绍了springcloud之自定义简易消费服务组件,本篇来使用rest+ribbon消费服务,并且通过轮询方式来自定义了个简易消费组件,感兴趣的小伙伴们可以参考一下
    2018-06-06
  • Java 切割字符串的几种方式集合

    Java 切割字符串的几种方式集合

    这篇文章主要介绍了Java 切割字符串的几种方式集合,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-09-09
  • Java如何实现图片的叠加与拼接操作

    Java如何实现图片的叠加与拼接操作

    这篇文章主要介绍了Java如何实现图片的叠加与拼接操作,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-11-11
  • java微信公众号开发案例

    java微信公众号开发案例

    这篇文章主要为大家详细介绍了java微信公众号开发案例,如何接入公众号,订阅号怎么样接收消息,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-11-11
  • mac下修改idea的jvm运行参数解决idea卡顿的情况

    mac下修改idea的jvm运行参数解决idea卡顿的情况

    这篇文章主要介绍了mac下修改idea的jvm运行参数解决idea卡顿的情况,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-12-12
  • 探讨如何在Eclipse中过滤版本控制文件.svn

    探讨如何在Eclipse中过滤版本控制文件.svn

    本篇文章是对在Eclipse中过滤版本控制文件.svn的方法进行了详细的分析介绍,需要的朋友参考下
    2013-07-07

最新评论