0

0

在Java中如何创建不可修改的集合_Java只读集合实现解析

P粉602998670

P粉602998670

发布时间:2026-01-28 10:37:02

|

407人浏览过

|

来源于php中文网

原创

Java中创建不可修改集合的三种主流方式是:Collections.unmodifiableXXX()系列(包装视图,不阻原始修改)、Java 9+ List.of()/Set.of()/Map.of()(直接构造不可变实例,禁null和重复)、Stream.toList()(Java 16+,返回新不可变列表)。

在java中如何创建不可修改的集合_java只读集合实现解析

Java中创建不可修改集合的三种主流方式

Java没有原生的“只读集合类型”,所有不可修改集合都是对已有集合的包装视图,底层仍依赖原始集合。一旦原始集合被修改,包装后的视图可能抛出UnsupportedOperationException,也可能出现不一致行为——这点极易被忽略。

最常用的是Collections.unmodifiableXXX()系列方法,它们返回的是即时快照式包装器;Java 9 引入的List.of()Set.of()Map.of()则直接构造不可变实例,更安全也更轻量。

  • Collections.unmodifiableList(list):适用于已有ArrayListLinkedList等可变集合,但需确保原始引用不再被使用
  • List.of("a", "b"):元素个数≤10时高效,内部用紧凑数组实现;超过10个元素会回退到ImmutableCollections.ListN
  • Arrays.asList(...).toArray()不能直接传给unmodifiableList()来“加固”——因为Arrays.asList()返回的是固定大小的列表,本身就不支持add()/remove(),但set()仍可用

为什么Collections.unmodifiableList()修改原始集合会导致问题

这个包装器不复制数据,只是拦截写操作并抛异常。但它**不阻止原始集合被其他引用修改**。比如:

List source = new ArrayList<>(Arrays.asList("x", "y"));
List unmod = Collections.unmodifiableList(source);
source.add("z"); // 合法!unmod.size() 现在是 3,遍历时会包含 "z"

此时unmod看似只读,实则已“脏”。这种隐式耦合是调试时最难定位的问题之一。

立即学习Java免费学习笔记(深入)”;

Figma
Figma

Figma 是一款基于云端的 UI 设计工具,可以在线进行产品原型、设计、评审、交付等工作。

下载
  • 根本原因:unmodifiableList()只是代理,不是副本
  • 修复思路:若需真正隔离,必须做深拷贝(如new ArrayList(source)再包装)
  • 注意Stream.toList()(Java 16+)返回的是不可修改列表,但它是新集合,与源无关联

Java 9+ of()系列方法的限制和陷阱

List.of()Set.of()Map.of()返回的是JVM内置的不可变实现,不仅禁止写操作,连null都不允许——这是和unmodifiableXXX()的关键区别

  • List.of("a", null) 直接抛 NullPointerException,而 Collections.unmodifiableList(Arrays.asList("a", null)) 是合法的
  • Set.of(1, 2, 2)IllegalArgumentException(重复元素),unmodifiableSet(new HashSet(Arrays.asList(1,2,2))) 不报错,但实际只存一个2
  • Map.of("k", "v") 最多支持10个键值对;更多请用Map.ofEntries(Map.entry(k1,v1), ...)

如何安全地对外暴露集合字段

类内部持有可变集合、但对外只提供只读视图,是常见设计需求。错误做法是每次getter都调用Collections.unmodifiableList(this.items)——这会反复创建新包装器,且无法防止多次调用后原始集合被改。

  • 推荐模式:在字段初始化时就构造不可变副本,例如 private final List items = List.of("a", "b");
  • 若必须动态构建,用new ArrayList(source) + Collections.unmodifiableList()组合,并确保source无外部引用
  • 避免返回this.items的任何可变视图,包括subList()stream().collect(Collectors.toList())等——它们可能绕过保护逻辑

真正的不可变性不在接口名里,而在对象生命周期和引用控制中。哪怕用了of(),如果把返回的List转成ArrayList再暴露出去,一切防护就失效了。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

443

2023.08.02

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

236

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

438

2024.03.01

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1102

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

189

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1566

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

18

2026.01.19

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

75

2025.09.05

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

158

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3万人学习

C# 教程
C# 教程

共94课时 | 7.8万人学习

Java 教程
Java 教程

共578课时 | 52.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号