Java中的不可变和不可修改Map

培训教学 潘老师 7个月前 (10-11) 336 ℃ (0) 扫码查看

在本教程中,我们将学习如何在Java中使用不可修改和不可修改的Map。不可变类有助于避免多线程应用程序中的许多设计挑战。

1.不可变与不可修改的Map

不支持修改操作的Map称为不可修改。不可修改的Map通常是其他可变Map的只读视图(包装器)。我们不能添加、删除或清除它们,但如果更改基础Map,此Map的视图也会更改。

Map<String, String> mutableMap = new HashMap<>();
mutableMap.put("key1", "value1");
Map<String, String> unmodifiableMap
    = Collections.unmodifiableMap(mutableMap);
//抛出 java.lang.UnsupportedOperationException
//unmodifiableMap.put("key2", "value2");
//更改在两个map中都可见
mutableMap.put("key2", "value2");
System.out.println(unmodifiableMap);  //{key1=value1, key2=value2}

不可变的Map保证底层Map对象永远不会可见任何更改。我们不能更改不可变的Map – 它们不包装另一个Map – 它们具有自己的元素。这些不是视图 – 这些是数据结构。其内容永远不会更改。

Map<String, String> immutableMap = Map.of("key1", "value1");
//抛出java.lang.UnsupportedOperationException
immutableMap.put("key2", "value2");

2.不可修改的Map

在Java 8中,引入了Collectors.unmodifiableMap(),作为Lambda表达式更改的一部分。这个静态工厂方法接受一个Map作为参数,并返回一个类型为java.util.Collections$UnmodifiableMap的不可修改视图。

Map<Integer, String> mutableMap = new HashMap<>();
//添加几个条目
Map<Integer, String> unmodifiableMap = Collections.unmodifiableMap(mutableMap);

Apache Commons Collections的MapUtils类也提供了类似的方法。

Map<String, String> unmodifiable = MapUtils.unmodifiableMap(mutableMap);

3.不可变的Map

3.1. 使用Map.of() – Java 9

Map.of()方法在Java 9中引入。使用此方法,我们可以创建包含零个或最多10个键值对的不可变Map。创建的Map属于java.util.ImmutableCollections$MapN类型,如果遇到任何null键或值,它将引发NullPointerException。

var unmodifiableMap = Map.of(1, "Mumbai", 2, "Pune", 3, "Bangalore");
var emptyUnmodifiableMap = Map.of();

3.2. 使用Guava的ImmutableMap

由于Guava是外部库,您需要将其添加到类路径中。如果使用Maven,可以像下面这样添加Guava依赖:

<dependency>
    <groupId>com.google.guava</groupId>
    <artifactId>guava</artifactId>
    <version>31.1-jre</version>
</dependency>

ImmutableMap是不可修改的Map实现。与其他不可修改的类类似,它拒绝null值。

可以通过以下方式创建ImmutableMap:

  • 使用copyOf方法
  • 使用of方法
  • 使用构建器

ImmutableMap.copyOf()接受一个Map作为输入参数,并创建包含与输入Map相似条目的不可修改Map。

Map<Integer, String> mutableMap = new HashMap<>();
mutableMap.put(1, "Mumbai");
mutableMap.put(2, "Pune");
mutableMap.put(3, "Bangalore");
var immutableMap = ImmutableMap.copyOf(mutableMap);

ImmutableMap.of()与Map.of()类似,只是它返回一个不可修改的Map,可以为空,也可以包含最多10个键值对。它返回类型为com.google.common.collect.RegularImmutableMap的实例。

var immutableMap = ImmutableMap.of(1, "Mumbai", 2, "Pune", 3, "Bangalore");
var emptyImmutableMap  = ImmutableMap.of();

ImmutableMap.builder()返回一个帮助创建不可修改Map的构建器。使用构建器,可以向不可修改Map添加原始底层Map中不存在的其他条目。

Map<Integer, String> mutableMap = new HashMap<>();
mutableMap.put(1, "Mumbai");
mutableMap.put(2, "Pune");
mutableMap.put(3, "Bangalore");
var immutableMap = ImmutableMap.builder()
    .putAll(mutableMap)
    .put(4, "Delhi")
    .build();

4.性能和效率

不可修改的Map返回原始Map的只读视图。它将是原始Map的薄代理。不可修改的Map比返回Map的副本要快得多,更节省内存。

然而,对原始Map的修改仍然会反映在不可修改的Map中。只有当没有人持有对原始Map的引用时,返回的Map才是真正不可修改的。

另一方面,不可修改的Map会创建原始Map的高效副本。当我们不希望修改Map或期望Map保持不变时,最好将其防御性地复制到不可修改的Map中。它保证一旦创建,不可修改的Map将不会发生修改,即使底层Map发生更改。

创建防御性副本可能略显昂贵。因此,如果我们有一个性能关键的应用程序,可能希望使用不可修改的Map。然而,如果我们希望确保Map保持不可修改,并且对底层Map的任何修改不会在应用程序中创建不一致,特别是在多线程环境中,可能要选择不可修改的Map。

5.结论

本Java教程探讨了创建不可变和不可修改Map的各种方法。建议使用我们正在使用的最新Java版本中可用的解决方案。


版权声明:本站文章,如无说明,均为本站原创,转载请注明文章来源。如有侵权,请联系博主删除。
本文链接:https://www.panziye.com/teach/9449.html
喜欢 (0)
请潘老师喝杯Coffee吧!】
分享 (0)
用户头像
发表我的评论
取消评论
表情 贴图 签到 代码

Hi,您需要填写昵称和邮箱!

  • 昵称【必填】
  • 邮箱【必填】
  • 网址【可选】