0

0

在Java里什么是不可变对象_Java安全设计思想解析

P粉602998670

P粉602998670

发布时间:2026-02-10 09:43:37

|

173人浏览过

|

来源于php中文网

原创

Java中不可变对象是设计约定而非语法强制,需满足类final、字段private final、构造器初始化且不逸出、返回内部可变对象时深拷贝四个条件;String符合,StringBuilder不符合。

在java里什么是不可变对象_java安全设计思想解析

不可变对象(Immutable Object)在 Java 中不是语法层面的强制约束,而是一种设计约定:一旦创建,其状态(即所有字段的值)就无法被修改。这直接关系到线程安全、缓存可靠性与防御性编程——不是“用了就安全”,而是“必须满足条件才真正不可变”。

为什么 String 是典型不可变对象,但 StringBuilder 不是

String 的不可变性依赖三个关键设计:内部字符数组 valueprivate final 修饰;所有可能修改内容的方法(如 substring()concat())都返回新对象;没有对外暴露可修改内部状态的 public 方法。而 StringBuildervalue 字段虽为 final,但其引用的 char[] 数组内容可被 append()setCharAt() 等方法反复修改,因此它可变。

常见误判点:

  • 仅靠 final 修饰字段 ≠ 对象不可变(如 final List list = new ArrayList();,list 引用不可变,但内容可变)
  • 未深拷贝可变组件(如类中持有 final Date date,但外部仍可通过 date.setTime() 修改其状态)
  • 未阻止子类重写方法(若类未声明 final,子类可能添加 setter 或暴露内部可变对象)

手写不可变类的四个硬性条件

要让自定义类成为真正不可变对象,缺一不可:

DeepAI
DeepAI

为天生具有创造力的人提供的AI工具

下载

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

  • 类本身用 final 修饰,防止继承后被破坏封装
  • 所有字段必须是 private final
  • 构造器完成全部初始化,且不暴露 this 引用(避免逸出)
  • 任何返回内部可变对象(如 java.util.Dateint[]ArrayList)的方法,必须返回副本而非原引用(例如用 new Date(internalDate.getTime())Collections.unmodifiableList() 包装)

示例片段:

public final class ImmutablePoint {
    private final int x;
    private final int y;

    public ImmutablePoint(int x, int y) {
        this.x = x;
        this.y = y;
    }

    public int getX() { return x; }
    public int getY() { return y; }
    // 没有 setter,不返回可变内部状态
}

不可变对象在并发场景下的真实价值与限制

不可变对象天然线程安全,因为无共享可变状态,不需要同步、锁或 volatile——这点在高竞争读场景(如配置类、枚举常量、函数式数据结构)中能显著降低开销。但它不能解决所有并发问题:

  • 不可变对象的“创建过程”仍需线程安全(如用非线程安全的 builder 构建后再赋值,可能看到半初始化状态)
  • 多个不可变对象组合的逻辑操作(如 “先读 A 再读 B,要求 A 和 B 一致”)仍需额外同步,不可变本身不保证原子性
  • 过度使用会增加 GC 压力(频繁创建新对象),比如用 ImmutablePoint 做高频坐标更新,不如用可变对象 + 明确同步边界

真正难的不是“让字段 final”,而是识别哪些字段表面不可变、实则暗藏可变引用;也不是“要不要不可变”,而是判断当前场景是否需要它带来的安全契约——比如 DTO 传参适合不可变,而实时计算中间状态通常不适合。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

668

2023.08.02

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1519

2023.10.24

string转int
string转int

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

668

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

555

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

193

2025.08.29

C++中int的含义
C++中int的含义

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

205

2025.08.29

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

69

2025.10.23

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

539

2023.12.01

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

152

2026.02.06

热门下载

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

精品课程

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

共23课时 | 3.4万人学习

C# 教程
C# 教程

共94课时 | 9万人学习

Java 教程
Java 教程

共578课时 | 61.6万人学习

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

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