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,你的线程私有状态就乱了。

MagicLight AI
MagicLight AI

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 并发编程模型,从语言级特性出发,深入理解 goroutine、channel 与调度机制。结合工程实践,分析并发设计模式、性能瓶颈与资源控制策略,帮助将并发能力有效转化为稳定、可扩展的系统性能优势。

2

2026.02.27

Golang 高级特性与最佳实践:提升代码艺术
Golang 高级特性与最佳实践:提升代码艺术

本专题深入剖析 Golang 的高级特性与工程级最佳实践,涵盖并发模型、内存管理、接口设计与错误处理策略。通过真实场景与代码对比,引导从“可运行”走向“高质量”,帮助构建高性能、可扩展、易维护的优雅 Go 代码体系。

1

2026.02.27

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新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号