登录
首页 >  文章 >  java教程

final关键字在Java中用于修饰引用类型时,表示该变量的引用地址不可变,但对象内部内容可以修改。以下是具体区分方式:一、final修饰引用类型时的行为地址不可变(Referenceisfinal)final关键字用于引用类型(如类、数组、接口等)时,表示该变量不能指向另一个对象。一旦初始化后,该变量的引用(即内存地址)不能再改变。对象内容可变(Objectcontentismutable)虽然

时间:2026-04-16 14:27:47 117浏览 收藏

在Java中,`final`修饰引用类型时,真正锁定的是变量的引用地址(即它不能再指向其他对象),而非对象内部状态——这意味着你仍可自由修改对象的内容,如调用`append()`、`add()`或`setName()`等方法;这种“指针不可换、盒子可装修”的特性常被误解为对象整体不可变,而真正的不可变性需类自身配合:私有final字段、无setter、构造器初始化及防御性拷贝等设计缺一不可;理解这一关键区别,不仅能避免多线程隐患和逻辑bug,更是写出健壮、可维护代码的基础。

怎么区分final修饰引用类型时地址不可变但对象内容可变

final 修饰引用类型,锁住的是“谁”,不是“什么”:它只保证这个变量不能再指向别的对象(地址不可变),但不阻止你通过这个引用去修改对象内部的状态(内容可变)。

一、final 引用 = 锁死“指针”,不锁死“盒子”

可以把引用想象成一张写有门牌号的纸条,而对象是门牌号对应的真实房子:

  • final 修饰引用:相当于把这张纸条封进胶水里——你不能再往上面改写新门牌号(不能重新赋值);
  • 但房子本身没上锁:只要你知道门牌号,仍可以进门、换灯泡、搬家具(调用方法修改属性)。

二、代码一看就懂

比如下面这段 Java 代码:

List finalList = new ArrayList<>();
finalList.add("a"); // ✅ 允许:修改对象内容
finalList = new ArrayList<>(); // ❌ 编译错误:试图改变引用指向

再比如自定义类:

final Person p = new Person("张三");
p.setName("李四"); // ✅ 允许:修改对象内部字段
p = new Person("王五"); // ❌ 编译错误:不能重新赋值

三、想真正“不可变”,得靠对象自己配合

final 只是引用层的约束。要让对象内容也不变,需额外设计:

  • 类的所有字段用 private + final 修饰;
  • 不提供修改字段的 setter 方法;
  • 构造器完成全部初始化;
  • 如果含可变对象(如 List),返回副本而非原引用(防御性拷贝)。

StringLocalDate 这些 JDK 中的“真正不可变类”,就是这么做的。

四、常见误区提醒

  • 误以为 final List list = new ArrayList<>(); 能防止添加元素——其实完全能 add/remove;
  • 混淆“引用不可变”和“对象不可变”,后者是更严格的设计目标;
  • 在多线程中,仅靠 final 引用不能保证线程安全——除非对象本身不可变或同步充分。

本篇关于《final关键字在Java中用于修饰引用类型时,表示该变量的引用地址不可变,但对象内部内容可以修改。以下是具体区分方式:一、final修饰引用类型时的行为地址不可变(Referenceisfinal)final关键字用于引用类型(如类、数组、接口等)时,表示该变量不能指向另一个对象。一旦初始化后,该变量的引用(即内存地址)不能再改变。对象内容可变(Objectcontentismutable)虽然引用地址固定,但对象本身的内容(字段值)是可以被修改的。二、示例代码finalStringBuildersb=newStringBuilder("Hello");sb.append("World");//✅可以修改对象内容System.out.println(sb);//输出:HelloWorld//sb=newStringBuilder("New");//❌编译错误:不能重新赋值三、与基本类型的对比|类型|是否可修改内容|是否可重新赋值||------------|----------------|----------------||finalint|❌不可修改|❌不可重新赋值||finalString|❌不可修改|❌不可重新赋值|》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于文章的相关知识,请关注golang学习网公众号!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>