Java中的“ final”关键字如何工作?(我仍然可以修改对象)

简介: Java中的“ final”关键字如何工作?(我仍然可以修改对象)

如何解决Java中的“ final”关键字如何工作?(我仍然可以修改对象)?
你总是允许初始化一个final变量。编译器确保你只能执行一次。

请注意,对存储在final变量中的对象的调用方法与的语义无关final。换句话说:final仅与引用本身有关,而与引用对象的内容无关。

Java没有对象不变性的概念。这是通过精心设计对象来实现的,这是一项艰巨的工作。

这是一个最喜欢的面试问题。带着这些问题,访问者试图找出你对对象的构造,方法,类变量(静态变量)和实例变量的理解程度。

import java.util.ArrayList;
import java.util.List;
class Test {

private final List foo;
public test() {
    foo = new ArrayList();
    foo.add("foo"); // Modification-1
}
public void setFoo(List foo) {
   //this.foo = foo; Results in compile time error.
}

}
在上述情况下,我们为“ Test”定义了一个构造函数,并为其指定了“ setFoo”方法。

关于构造器: 构造函数可以调用只有一个使用每个对象的创建时间new关键字。你不能多次调用构造函数,因为构造函数的目的不是这样做的。

关于方法:可以根据需要多次调用方法(甚至永远不会调用),并且编译器知道该方法。

场景1

private final List foo; // 1
foo是实例变量。当我们创建Test类对象时,实例变量foo将被复制到Test类对象中。如果我们foo在构造函数内部分配,则编译器知道构造函数将仅被调用一次,因此在构造函数内部分配它没有问题。

如果我们foo在一个方法内部赋值,编译器就会知道一个方法可以被多次调用,这意味着必须多次更改该值,而final变量是不允许的。因此编译器认为构造函数是一个不错的选择!你只能一次将值分配给最终变量。

方案2

private static final List foo = new ArrayList();
foo现在是一个静态变量。当我们创建Test类的实例时,foo不会被复制到对象,因为它foo是静态的。现在foo不是每个对象的独立属性。这是Test类的属性。但是foo可以被多个对象看到,如果每个使用new关键字创建的对象最终将调用Test构造函数,则该构造函数会在创建多个对象时更改值(记住static foo并不是在每个对象中都复制,而是在多个对象之间共享)

场景3

t.foo.add("bar"); // Modification-2
以上Modification-2是你的问题。在上述情况下,你没有更改第一个引用的对象,而是在foo其中添加了允许的内容。如果你尝试将a分配new ArrayList()给foo引用变量,则编译器会抱怨。 规则如果已初始化final变量,则无法将其更改为引用其他对象。(在这种情况下ArrayList)

final类不能被子类化final方法不能被覆盖。(此方法在超类中) 最终方法可以重写。(以语法方式阅读。此方法在子类中)

解决方法
在Java中,我们使用final带有变量的关键字来指定其值不被更改。但是我看到你可以在类的构造函数/方法中更改值。同样,如果变量是static,则为编译错误。

这是代码:

import java.util.ArrayList;
import java.util.List;
class Test {
private final List foo;
public Test()
{

  foo = new ArrayList();
  foo.add("foo"); // Modification-1

}
public static void main(String[] args)
{

  Test t = new Test();
  t.foo.add("bar"); // Modification-2
  System.out.println("print - " + t.foo);

}
}
上面的代码工作正常,没有错误。

现在将变量更改为static:

private static final List foo;
现在是编译错误。这final真的如何运作?
声明:地推任务网所有作品(图片、文字)均由用户自行上传分享,仅供网友学习交流。

相关文章
|
9天前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
13天前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
35 17
|
13天前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
22天前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第7天】Java零基础教学篇,手把手实践教学!
23 6
|
23天前
|
Java 程序员
Java 面试高频考点:static 和 final 深度剖析
本文介绍了 Java 中的 `static` 和 `final` 关键字。`static` 修饰的属性和方法属于类而非对象,所有实例共享;`final` 用于变量、方法和类,确保其不可修改或继承。两者结合可用于定义常量。文章通过具体示例详细解析了它们的用法和应用场景。
25 3
|
21天前
|
存储 安全 Java
了解final关键字在Java并发编程领域的作用吗?
在Java并发编程中,`final`关键字不仅用于修饰变量、方法和类,还在多线程环境中确保对象状态的可见性和不变性。本文深入探讨了`final`关键字的作用,特别是其在final域重排序规则中的应用,以及如何防止对象的“部分创建”问题,确保线程安全。通过具体示例,文章详细解析了final域的写入和读取操作的重排序规则,以及这些规则在不同处理器上的实现差异。
了解final关键字在Java并发编程领域的作用吗?
|
12天前
|
存储 缓存 NoSQL
一篇搞懂!Java对象序列化与反序列化的底层逻辑
本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。
10 0
|
24天前
|
存储 前端开发 Java
你还没有对象吗?java带你创建一个吧
你还没有对象吗?java带你创建一个吧
10 0
|
2月前
|
存储 Java
Java编程中的对象和类
【8月更文挑战第55天】在Java的世界中,“对象”与“类”是构建一切的基础。就像乐高积木一样,类定义了形状和结构,而对象则是根据这些设计拼装出来的具体作品。本篇文章将通过一个简单的例子,展示如何从零开始创建一个类,并利用它来制作我们的第一个Java对象。准备好让你的编程之旅起飞了吗?让我们一起来探索这个神奇的过程!
30 10
|
2月前
|
存储 Java
Java的对象和类的相同之处和不同之处
在 Java 中,对象和类是面向对象编程的核心。