0

0

并发编程中的防御性拷贝_保护线程内部私有状态不被外部修改

P粉602998670

P粉602998670

发布时间:2026-02-27 12:35:02

|

660人浏览过

|

来源于php中文网

原创

clone() 不一定能防住外部修改,因为其默认为浅拷贝,仅复制引用地址而非实际内容;含可变引用(如arraylist)时需手动深拷贝,数组宜用arrays.copyof(),集合返回应避免直接暴露内部引用。

并发编程中的防御性拷贝_保护线程内部私有状态不被外部修改

为什么 clone() 不一定能防住外部修改

因为很多对象的 clone() 是浅拷贝——只复制了引用类型字段的地址,没复制实际内容。比如你有个 ArrayList<string></string> 字段,clone() 后外部仍能通过原列表或副本往里 add 元素,影响彼此。

  • 原始对象含可变引用(如 ArrayListHashMap、自定义可变类)时,必须手动深拷贝关键字段
  • Arrays.copyOf() 替代 clone() 处理数组更可控,尤其对基本类型数组或不可变元素数组
  • Java 9+ 可考虑 List.copyOf()Set.copyOf(),但注意它们返回的是不可变视图,不是新容器;若需可变副本,得用 new ArrayList(original)

构造函数里怎么安全接收外部集合参数

常见错误是直接把传进来的 List 赋给内部字段:this.items = items; —— 这等于把控制权交出去了,调用方随时能改它。

  • 始终用防御性初始化:例如 this.items = new ArrayList(items);,前提是 items 元素本身不可变(如 StringInteger
  • 如果元素可变(比如 Person 对象),且你不希望外部修改影响内部状态,就得逐个深拷贝:this.items = items.stream().map(Person::new).collect(Collectors.toList());
  • 别依赖 Collections.unmodifiableList() 做防御——它只是加了运行时检查,不阻止原始引用被修改,也不能防止传入前就被共享

返回内部集合时为什么不能直接 return this.list

这是最常踩的坑:方法返回 this.dataList,调用方拿到后一通 add/clear,你的线程私有状态就乱了。

Spell.tools
Spell.tools

高颜值AI内容营销创作工具

下载
  • 永远返回副本:return new ArrayList(this.dataList);(元素不可变时)
  • 如果元素可变且需隔离,返回不可变包装 + 深拷贝组合:return Collections.unmodifiableList(this.dataList.stream().map(Widget::copy).toList());
  • 避免用 Arrays.asList() 包装数组后返回——它返回的是固定大小的 List,底层仍直连原数组,且不防写操作

用 record 或不可变类简化防御逻辑

Java 14+ 的 record 天然不可变,字段自动 final,没有 setter,构造即完成封装。但它不解决“字段本身是否可变”的问题。

  • record Config(String name, List<string> tags) {}</string> 看似安全,但 tags 仍可被外部修改——record 只保证引用不变,不保证内容不变
  • 正确做法:在 record 构造器里做防御拷贝:record Config(String name, List<string> tags) { this(name, new ArrayList(tags)); }</string>
  • 如果字段是数组,record 会直接暴露引用,必须用 Arrays.copyOf() 或包装为 List 再封装

真正难的不是拷贝动作本身,而是判断哪些字段需要深拷贝、哪些可以浅拷贝、哪些其实该设计成不可变。很多人卡在“不知道该在哪一层切断引用”,结果要么过度拷贝拖慢性能,要么漏掉一个可变字段导致偶发并发异常。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

870

2023.08.02

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

721

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

721

2023.08.10

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

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

77

2025.09.05

golang map相关教程
golang map相关教程

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

38

2025.11.16

golang map原理
golang map原理

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

67

2025.11.17

java判断map相关教程
java判断map相关教程

本专题整合了java判断map相关教程,阅读专题下面的文章了解更多详细内容。

47

2025.11.27

Golang 实际项目案例:从需求到上线
Golang 实际项目案例:从需求到上线

《Golang 实际项目案例:从需求到上线》以真实业务场景为主线,完整覆盖需求分析、架构设计、模块拆分、编码实现、性能优化与部署上线全过程,强调工程规范与实践决策,帮助开发者打通从技术实现到系统交付的关键路径,提升独立完成 Go 项目的综合能力。

2

2026.02.26

Golang Web 开发路线:构建高效后端服务
Golang Web 开发路线:构建高效后端服务

《Golang Web 开发路线:构建高效后端服务》围绕 Go 在后端领域的工程实践,系统讲解 Web 框架选型、路由设计、中间件机制、数据库访问与接口规范,结合高并发与可维护性思维,逐步构建稳定、高性能、易扩展的后端服务体系,帮助开发者形成完整的 Go Web 架构能力。

5

2026.02.26

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.5万人学习

Rust 教程
Rust 教程

共28课时 | 6.3万人学习

Django 教程
Django 教程

共28课时 | 4.6万人学习

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

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