0

0

在Java里final关键字的作用是什么_Java不变性设计解析

P粉602998670

P粉602998670

发布时间:2026-02-23 11:17:02

|

857人浏览过

|

来源于php中文网

原创

final修饰变量时锁住的是变量的“引用指向”而非对象内容;基本类型值固定,引用类型仅限指向初始对象;需配合不可变类才能实现真正不可变。

在java里final关键字的作用是什么_java不变性设计解析

final修饰变量时到底锁住了什么

它锁住的是变量的“引用指向”,不是对象内容。比如 final List<string> list = new ArrayList();</string> 后,你不能执行 list = new ArrayList();(编译报错),但可以调用 list.add("a"); —— 因为 ArrayList 本身是可变的。

常见误判:以为加了 final 就等于“不可变对象”。其实只是防止重新赋值,和对象内部状态无关。

  • 基本类型变量(如 intboolean)加 final 后值彻底固定
  • 引用类型变量加 final 后只能指向初始化时的那个对象实例
  • 若需真正不可变,得配合不可变类(如 ImmutableList)或手动封装

final方法为什么不能被重写但能被重载

final 方法禁止子类覆盖实现,但不影响同名不同参的重载。这是继承机制里的明确限制:JVM 在编译期就拒绝生成对 final 方法的 invokespecial 覆盖调用。

典型场景:模板方法模式中,把骨架逻辑用 final 封装,只开放钩子方法给子类实现。

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

Rezi.ai
Rezi.ai

一个使用 AI 自动化创建简历平台

下载
  • 子类定义同签名方法会直接编译失败,错误信息类似:Cannot override the final method from XXX
  • 子类定义同名但参数不同的方法是合法的,属于重载,与 final 无关
  • 注意反射绕过:虽然 setAccessible(true) 可访问 final 字段,但无法通过反射“重写” final 方法

final类在序列化和代理场景下的实际影响

final 类不能被继承,因此无法被 CGLIB 动态代理(CGLIB 靠生成子类实现),但不影响 JDK 动态代理(基于接口)和 Jackson/Gson 序列化(只要字段可读/可写)。

如果你用 Spring AOP 默认配置对 final 类加切面,会静默退化为 JDK 代理——前提是该类实现了接口;否则切面不生效,且无任何提示。

  • Jackson 反序列化 final 字段需要显式构造器或 @JsonCreator,否则忽略该字段
  • Gson 默认跳过 final 字段(除非开启 setExclusionStrategies 或用 @SerializedName 配合构造器)
  • Lombok 的 @Data@Value 行为不同:@Value 默认生成 final 字段 + 不可变构造器,@Data 不加 final

final字段的线程安全边界在哪

只有满足“正确构造”的 final 字段,才能保证对其他线程的可见性。所谓正确构造,是指构造过程中没有发生“this逃逸”——即构造函数结束前,this 引用未被发布(如未传给其他线程、未注册到静态容器、未启动内部线程等)。

一旦发生逃逸,即使字段是 final,其他线程仍可能看到默认值(如 null0)。

  • JMM 保证:线程 A 构造完对象后,线程 B 第一次读取该对象的 final 字段,一定能看到构造器中写入的值(前提是没有逃逸)
  • 逃逸例子:new Thread(() -> System.out.println(this)).start(); 放在构造函数里
  • 不要依赖 final 实现线程安全的“完全体”——它只解决发布时的可见性,不解决后续读写竞争
真正容易被忽略的是:final 的语义保障全靠 JVM 和编译器协同,而逃逸检测没有任何运行时检查。写错一次,问题可能只在高并发压测时才浮现。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

143

2025.08.06

Java Spring Security 与认证授权
Java Spring Security 与认证授权

本专题系统讲解 Java Spring Security 框架在认证与授权中的应用,涵盖用户身份验证、权限控制、JWT与OAuth2实现、跨站请求伪造(CSRF)防护、会话管理与安全漏洞防范。通过实际项目案例,帮助学习者掌握如何 使用 Spring Security 实现高安全性认证与授权机制,提升 Web 应用的安全性与用户数据保护。

81

2026.01.26

java中boolean的用法
java中boolean的用法

在Java中,boolean是一种基本数据类型,它只有两个可能的值:true和false。boolean类型经常用于条件测试,比如进行比较或者检查某个条件是否满足。想了解更多java中boolean的相关内容,可以阅读本专题下面的文章。

364

2023.11.13

java boolean类型
java boolean类型

本专题整合了java中boolean类型相关教程,阅读专题下面的文章了解更多详细内容。

39

2025.11.30

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

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

246

2023.09.22

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

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

846

2024.03.01

string转int
string转int

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

830

2023.08.02

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

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

580

2024.08.29

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

1127

2026.02.13

热门下载

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

精品课程

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

共23课时 | 3.8万人学习

C# 教程
C# 教程

共94课时 | 10万人学习

Java 教程
Java 教程

共578课时 | 70.9万人学习

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

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