Java 中的final:不可变性的魔法之旅
🎏:你只管努力,剩下的交给时间
🏠 :小破站
Java 中的final:不可变性的魔法之旅
- 前言
- 第一:了解final变量
- 第二:final方法
- 第三:final类
- 第四:final参数
- 第五:final引用和对象
- 第六:final和多线程
- 第七:final和性能优化
前言
在 Java 编程世界中,final 是一个引人注目的关键字,它赋予了变量、方法、类等各种元素不可变性。有些程序员将其视为一种约束,而另一些则将其视为一种保护措施。在这个博客中,我们将探索final的多种用法,从变量的不可变性到类的终结,了解其妙用。final是你代码的最后一道屏障,让我们一起发现它的力量。
第一:了解final变量
final变量是一种Java中的变量类型,表示一旦被赋值就不能再次修改。final可以用于变量、方法和类,它提供了不可变性和确定性的特性,有助于代码的可靠性和安全性。
以下是关于final变量的声明、初始化以及不可变性的意义和好处的详细信息:
声明和初始化final变量:
-
final变量:在Java中,你可以使用final关键字声明一个不可变的变量。一旦给final变量赋值,它就不能再次被修改。例如:final int age = 30; -
final方法:在方法声明中使用final关键字表示该方法不能被子类覆盖(重写)。这可以用于确保特定方法的行为在子类中不会被改变。public final void myFinalMethod() {// 方法内容 } -
final类:在类声明中使用final关键字表示该类不能被继承。这用于创建不可被继承的最终类。public final class MyFinalClass {// 类内容 }
不可变性的意义和好处:
-
线程安全:不可变对象在多线程环境中是线程安全的,因为它们的状态不会改变。这消除了需要使用同步机制的复杂性。
-
简化代码:不可变对象更容易理解,因为它们的状态不会改变。这可以减少代码中的复杂性,提高可维护性。
-
缓存:不可变对象适合用作缓存键,因为它们的哈希值在其生命周期内不会改变。
-
性能优化:编译器和运行时环境可以对不可变对象进行优化,以提高性能。
-
可靠性:不可变对象不容易受到错误的影响,因为它们不会在运行时修改状态。
-
可重用性:不可变对象可以被多个线程共享,从而提高了对象的可重用性。
-
安全性:不可变对象在安全性上有优势,因为它们不容易受到恶意篡改。
总之,final变量用于创建不可变对象,具有不可变性的对象在多线程环境中更安全,更容易理解,能够提供更好的性能和可维护性。它们在编写高质量Java代码时具有重要的作用。
第二:final方法
final方法是Java中的一种方法修饰符,用于表示该方法不能被子类覆盖(重写)。一旦一个方法被声明为final,它在子类中不能被修改或覆盖。这是为了确保方法的行为不会被改变,增强代码的可靠性和稳定性。
以下是关于final方法的声明以及方法覆盖的规则和限制的详细信息:
声明不可覆盖的方法:
在类中声明不可覆盖的方法需要使用final关键字。以下是一个示例:
public class ParentClass {public final void myFinalMethod() {// 方法内容}
}
在上述示例中,myFinalMethod方法被声明为final,表示它不能被子类修改或覆盖。
方法覆盖的规则和限制:
在Java中,方法覆盖(也称为方法重写)是一种子类重写父类的方法,以提供自己的实现。以下是有关方法覆盖的规则和限制:
-
子类方法必须具有与父类方法相同的名称、参数列表和返回类型。
-
子类方法不能缩小父类方法的访问权限。例如,如果父类方法是
public,则子类方法不能是private。 -
子类方法不能抛出比父类方法更多的异常。子类方法可以不抛出异常或抛出父类方法抛出的异常。
-
final方法不能被覆盖。 -
静态方法不能被覆盖,因为它们与类而不是实例相关联。
-
构造方法不能被覆盖。
-
子类方法的返回类型可以是父类方法返回类型的子类(协变返回类型),但不能是父类方法返回类型的超类。
方法覆盖是面向对象编程中的一个重要概念,它允许子类提供自己的实现以满足特定需求,同时保留了父类的接口。使用final方法可以禁止方法的覆盖,以确保特定的行为不会被改变。
第三:final类
final类是Java中的一种类修饰符,用于表示该类不能被继承。一旦一个类被声明为final,它不允许有子类,确保该类的实现不会被修改。这是为了增强类的稳定性和可靠性。
以下是有关final类的声明以及类继承和扩展思考的详细信息:
声明不可继承的类:
在Java中,要声明一个不可继承的类,只需在类的声明前加上final关键字。以下是一个示例:
public final class MyFinalClass {// 类内容
}
在上述示例中,MyFinalClass被声明为final,表示它不能有任何子类。
类继承和扩展的思考:
在面向对象编程中,类继承和扩展是一个重要的概念,允许你创建新的类,从已有的类继承属性和行为。然而,有时你可能希望限制某个类的继承,或者确保某个类的实现不会被修改。以下是一些考虑类继承和扩展的思考:
-
继承的合适性:在考虑是否继承某个类时,你应该考虑是否确实需要继承,是否需要重用父类的行为,以及是否符合面向对象设计原则。
-
类的设计:当你设计一个类时,要考虑它是否应该允许继承。有些类被设计为基类,允许其他类继承并扩展它们的功能,而有些类可能被设计为不可继承,以确保其行为的稳定性。
-
final类的使用:final类通常用于表示某个类的实现是最终的,不应该被修改。这对于库类、不可变类或具有特定行为的类非常有用。 -
接口的使用:如果你想要提供一种方式来共享某个类的行为而不涉及继承,可以考虑使用接口。接口允许多个类实现共享的行为。
总之,final类用于表示不可继承的类,它可以确保类的实现稳定性和不可修改性。在设计和扩展类时,要仔细考虑继承和扩展的需求,并根据需要使用final类或接口来实现设计目标。
第四:final参数
在方法参数中使用final的主要目的是为了增加参数的不可修改性(immutability)和提高代码的安全性。当你将一个参数声明为final时,它在方法内部不能被修改,这有助于防止意外的更改和提高代码的可维护性。
以下是有关在方法参数中使用final的详细信息以及示例:
为什么在方法参数中使用final:
-
不可修改性:使用
final参数可以确保方法内部不会更改参数的值。这有助于防止在方法中无意间修改参数的值,导致不可预测的行为。 -
安全性:不可修改的参数可以提高代码的安全性,特别是在多线程环境中。多线程环境下,共享的可修改参数可能导致竞态条件和数据竞争。通过使用
final参数,可以避免这些问题。 -
可读性:
final参数可以增加代码的可读性,因为它明确表明这个参数在方法中是只读的,不会被修改。
示例:在方法中使用final参数的情况:
public class Example {public void process(final int number) {// 使用final参数System.out.println("Received number: " + number);// 以下代码会导致编译错误,因为number是final,不能被修改// number = number * 2; // 编译错误}public static void main(String[] args) {Example example = new Example();int value = 42;example.process(value);System.out.println("Value after method call: " + value);}
}
在上述示例中,process方法接受一个final参数number,并在方法内部使用它。由于number被声明为final,任何尝试在方法内部修改它的操作都会导致编译错误。这确保了number的不可修改性和代码的安全性。在方法调用后,value的值仍然保持不变,没有被修改。
通过在方法参数中使用final,你可以增加代码的可读性和安全性,确保参数的不可修改性,减少潜在的错误。这在编写高质量Java代码时是一个有用的实践。
第五:final引用和对象
理解final引用和final对象之间的区别是非常重要的,它们涉及到Java中的不可变性和可变性的概念。以下是这两者之间的区别以及不可变对象的优势:
Final引用 vs. Final对象:
-
Final引用:当你将一个引用声明为
final时,意味着这个引用不能再指向其他对象。但它并不影响引用所指对象的可变性。你仍然可以通过这个final引用修改对象的内部状态。final StringBuilder builder = new StringBuilder("Hello"); builder.append(", World"); // 合法,修改了builder所指对象的内部状态 -
Final对象:当你将一个对象声明为
final时,意味着这个对象自身不能被重新分配(即不允许重新创建对象并将引用指向其他对象),并且它的内部状态(如果有)也不能被修改。这确保了对象的不可变性。final String str = "Hello"; str = "World"; // 编译错误,不能重新分配str的引用
Final引用对于集合和数组的影响:
当引用是final时,它的不可变性仅限于引用自身,而不限于引用所指向的对象。这意味着,如果引用是final,你仍然可以修改引用所指向的对象的内部状态,包括集合和数组中的元素。例如:
final List<String> list = new ArrayList<>();
list.add("Hello"); // 合法,修改了list所指向的ArrayList对象
为了实现不可变的集合或数组,需要采取其他措施,如使用Collections.unmodifiableList或不提供修改方法的自定义不可变集合。
不可变对象的优势:
-
线程安全:不可变对象在多线程环境中是线程安全的,因为它们的状态不会改变,无需锁定。
-
缓存:不可变对象适合用作缓存键,因为它们的哈希值不会改变。
-
可靠性:不可变对象在运行时不会改变,这有助于避免意外的修改和错误。
-
性能优化:编译器和运行时环境可以对不可变对象进行优化,提高性能。
-
可重用性:不可变对象可以被多个线程共享,从而提高了对象的可重用性。
在Java中,不可变对象具有许多优势,因此在设计和编写代码时,考虑对象的不可变性是一个有益的实践。
第六:final和多线程
final在多线程环境下具有重要的作用,它主要用于确保多线程下的可见性和不可变性。以下是final在多线程环境中的作用、如何避免竞态条件和提高线程安全性以及使用final的最佳实践:
final在多线程环境下的作用:
-
可见性:当一个线程在构造对象时,如果将字段标记为
final,则其他线程在访问该对象时能够立即看到这些final字段的最新值,而不需要额外的同步机制。 -
不可变性:
final字段保证它们的值不会被修改,从而创建了不可变对象。不可变对象是线程安全的,因为它们不会在多线程环境中发生状态变化。
避免竞态条件和线程安全性:
-
不可变对象:使用
final关键字来确保对象的不可变性。不可变对象在多线程环境中是线程安全的,不需要额外的同步。 -
保证可见性:在多线程编程中,
final字段的值对其他线程可见。这可以帮助避免竞态条件,确保线程之间的协同工作。 -
使用
final的对象引用:如果你将一个对象引用声明为final,确保在多线程环境中,这个引用不会被修改。这有助于避免引用变量被重分配的问题。 -
不可变集合:Java提供了
Collections.unmodifiableXXX方法来创建不可变的集合对象。这些不可变集合在多线程环境中提供了线程安全性。
使用final的最佳实践:
-
尽量将字段声明为
final:在可能的情况下,将字段声明为final以提高代码的可读性和线程安全性。 -
使用
final局部变量:在方法中使用final局部变量,特别是在匿名内部类或Lambda表达式中,以确保它们不会发生意外的变化。 -
谨慎使用
volatile:volatile字段用于确保字段的可见性,但不一定保证不可变性。在需要不可变性的情况下,使用final字段更为安全。 -
避免线程逃逸:确保不会将
this引用传递给其他线程,以避免线程逃逸问题。
总之,final关键字在多线程编程中是一个有用的工具,可以帮助确保可见性和不可变性,从而提高线程安全性。它在不可变对象、字段、局部变量等方面都有不可替代的作用,但仍需在设计和编写多线程代码时谨慎使用。
第七:final和性能优化
final关键字与性能优化密切相关,因为它为编译器提供了有关代码优化的信息。以下是final与编译器优化、内联和常量折叠之间的关系,以及final对性能的影响:
编译器优化和final的关系:
-
不变性保证:当一个字段或变量被声明为
final时,编译器知道它的值不会在运行时发生变化。这允许编译器执行一些优化,如不必要的内存访问和计算的消除。 -
方法内联:编译器可能会选择将
final方法内联到调用它的地方,从而避免了方法调用的开销。 -
常量折叠:编译器可以执行常量折叠,将
final常量的值在编译时直接替换到使用它的地方。
内联和常量折叠:
-
内联:内联是一种优化技术,它将方法调用处的代码替换为被调用方法的实际内容,从而减少方法调用的开销。在某些情况下,编译器可能会选择内联
final方法,因为它知道这些方法不会在子类中被覆盖。 -
常量折叠:常量折叠是指编译器将编译时已知的常量的计算结果直接插入到代码中。
final常量通常可以被折叠,因为它们的值在编译时已知并且不可变。
final对性能的影响:
-
性能提升:
final常量的使用通常有助于提高性能,因为它们可以被编译器优化,以减少运行时开销。 -
减少不必要的内存访问:
final字段通常不需要被读取多次,因为它们的值不会改变。这可以减少不必要的内存访问,提高代码的效率。 -
代码可读性:
final可以提高代码的可读性,因为它明确表明某个字段或方法是不可变的,不会在运行时改变。
虽然final可以提高性能,但应该根据具体情况来决定是否使用它。在某些情况下,过度使用final可能会导致不必要的复杂性,因此需要谨慎使用。通常情况下,使用final常量和final方法是一种良好的实践,可以提高代码的可读性和性能。
相关文章:
Java 中的final:不可变性的魔法之旅
🎏:你只管努力,剩下的交给时间 🏠 :小破站 Java 中的final:不可变性的魔法之旅 前言第一:了解final变量第二:final方法第三:final类第四:final参数第五&#…...
Alfred 5 for mac(最好用的苹果mac效率软件)中文最新版
Alfred 5 Mac是一款非常实用的工具,它可以帮助用户更加高效地使用Mac电脑。用户可以学会使用快捷键、全局搜索、快速启动应用程序、使用系统维护工具、快速复制粘贴文本以及自定义设置等功能,以提高工作效率。 Alfred for Mac 的一些主要功能包括&#…...
常见的Python解释器,你了解多少?
Python,作为一种解释型编程语言,它的运行过程也遵循“程序源码—>解释器(字节码)—>虚拟机(可执行文件)”的流程。 在编写Python程序时,是在扩展名为.py的文件中进行编写,.py…...
在 Python 中使用 Selenium 按文本查找元素
我们将通过示例介绍在Python中使用selenium通过文本查找元素的方法。 在 Python 中使用 Selenium 按文本查找元素 软件测试是检查应用程序是否满足用户需求的技术。 该技术有助于使应用程序成为无错误的应用程序。 软件测试可以手动完成,也可以通过某些软件完成。…...
【Notepad++】搜索返回窗口(find result)被隐藏或遮挡如何恢复?
Notepad 搜索返回窗口被隐藏或遮挡如何恢复 1:F72:F12恢复之后可以多看一些Notepad中快捷键的使用,以备不时之需。 1:F7 打开任意文件,搜索任意内容,按F7,焦点切换到Find result。 按AltSpace,出现小窗口点击"移动…...
应用软件安全编程--05预防 XML 注入
如果用户有能力使用结构化XML 文档作为输入,那么他能够通过在数据字段中插入 XML 标签来 重写这个 XML 文档的内容。 XML 解析器会将这些标签按照正常标签进行解析。下面是一段在线商 店的 XML 代码,主要用于查询后台数据库。 <item)<descri…...
JavaEE-博客系统3(功能设计)
本部分内容为:实现登录功能;强制要求用户登录;实现显示用户信息;退出登录;发布博客 该部分的后端代码如下: Overrideprotected void doPost(HttpServletRequest req, HttpServletResponse resp) throws Ser…...
椭圆滤波器
之前的文章 信号去噪 中列出了7种常用的信号去噪算法,对于后两种算法——深度学习和奇异值分解(SVD),我现在也不太理解,就先不写了。 很多朋友留言又提了一些算法,今天一起来聊聊椭圆滤波器。 椭圆滤波器(Elliptic F…...
Mac 下安装golang环境
一、下载安装包 安装包下载地址 下载完成,直接继续----->下一步到结束即可安装成功; 安装成功之后,验证一下; go version二、配置环境变量 终端输入vim ~/.zshrc进入配置文件,输入i进行编辑 打开的不管是空文本…...
前端面试大纲
一、CSS 1.说一下CSS的盒模型。 在HTML页面中的所有元素都可以看成是一个盒子 盒子的组成:内容content、内边距padding、边框border、外边距margin 盒模型的类型: 标准盒模型 margin border padding content IE盒模型 margin content(border padd…...
CAN(Controller Area Network)是一种用于在汽车和工业领域中进行通信的串行总线系统(附加案例)
CAN(Controller Area Network)是一种用于在汽车和工业领域中进行通信的串行总线系统。它是一种高可靠性、多主机、多节点通信协议,主要用于实时控制和数据传输。 CAN数据是指在CAN总线上通过CAN协议进行通信传输的数据。CAN总线上的数据被分…...
代码随想录day53|1143.最长公共子序列、 1035.不相交的线、 53. 最大子序和
1143.最长公共子序列 dp[i][j]:长度为[0, i - 1]的字符串text1与长度为[0, j - 1]的字符串text2的最长公共子序列为dp[i][j] 因此是if(nums1[i-1] nums2[j-1]) 1035.不相交的线 和上一题一样 53. 最大子序和 int result dp[0]; 不是0,因为dp[i]有…...
xilinx fpga ddr mig axi
硬件 参考: https://zhuanlan.zhihu.com/p/97491454 https://blog.csdn.net/qq_22222449/article/details/106492469 https://zhuanlan.zhihu.com/p/26327347 https://zhuanlan.zhihu.com/p/582524766 包括野火、正点原子的资料 一片内存是 1Gbit 128MByte 16bit …...
《golang设计模式》第三部分·行为型模式-04-迭代器模式(Iterator)
文章目录 1. 概念1.1 角色1.2 类图 2. 代码示例2.1 需求2.2 代码2.3 类图 1. 概念 迭代器(Iterator)能够在不暴露聚合体内部表示的情况下,向客户端提供遍历聚合元素的方法。 1.1 角色 InterfaceAggregate(抽象聚合)…...
python加上ffmpeg实现音频分割
前言: 这是一个系列的文章,主要是使用python加上ffmpeg来对音视频文件进行处理,包括音频播放、音频格式转换、音频文件分割、视频播放等。 系列文章链接: 链接1: python使用ffmpeg来制作音频格式转换工具(优化版) 链接2:<Python>PyQt5+ffmpeg,简单视频播放器的编写(…...
LLM之Prompt(一):5个Prompt高效方法在文心一言3.5的测试对比
在Effective Prompt: 编写高质量Prompt的14个有效方法文中我们了解了14个编写Prompt的方法(非常感谢原作者),那么这些Prompt在具体大模型中的效果如何呢?本文以百度文心一言3.5版本大模型在其中5个方法上做个测试对比。 第1条&am…...
TreeBERT:基于树的编程语言预训练模型。
TreeBERT https://arxiv.org/abs/2105.12485 Comments: Accepted by UAI2021 Subjects: Machine Learning (cs.LG); Programming Languages (cs.PL) Cite as: arXiv:2105.12485 [cs.LG] 1 Introduction 现有挑战: 设计适当的机制来学习程序的语法结构 代码是强结…...
生成小程序的二维码的base64码(中间logo可以自定义)
1.生成基础二维码 /*** 生成微信小程序二维码,带参数,最终转成base64* param page 当前小程序相对页面 必须是已经发布的小程序存在的页面(否则报错),例如 pages/index/index, 根路径前不要填加 /,不能携带参数(参数请放在scene字段里),如果不…...
【音视频 | Ogg】Ogg封装格式详解——包含Ogg封装过程、数据包(packet)、页(page)、段(segment)等
😁博客主页😁:🚀https://blog.csdn.net/wkd_007🚀 🤑博客内容🤑:🍭嵌入式开发、Linux、C语言、C、数据结构、音视频🍭 🤣本文内容🤣&a…...
ubuntu 22.04 安装ros2 iron
目录 一、镜像 二、ifconfig 三、ssh连接不上 四、安装iron 一、镜像 123云盘直接下载 二、ifconfig sudo apt install net-tools 三、ssh连接不上 sudo apt install openssh-server 四、安装iron Ubuntu (Debian packages) — ROS 2 Documentation: Iron document…...
(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)
题目:3442. 奇偶频次间的最大差值 I 思路 :哈希,时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况,哈希表这里用数组即可实现。 C版本: class Solution { public:int maxDifference(string s) {int a[26]…...
挑战杯推荐项目
“人工智能”创意赛 - 智能艺术创作助手:借助大模型技术,开发能根据用户输入的主题、风格等要求,生成绘画、音乐、文学作品等多种形式艺术创作灵感或初稿的应用,帮助艺术家和创意爱好者激发创意、提高创作效率。 - 个性化梦境…...
19c补丁后oracle属主变化,导致不能识别磁盘组
补丁后服务器重启,数据库再次无法启动 ORA01017: invalid username/password; logon denied Oracle 19c 在打上 19.23 或以上补丁版本后,存在与用户组权限相关的问题。具体表现为,Oracle 实例的运行用户(oracle)和集…...
java 实现excel文件转pdf | 无水印 | 无限制
文章目录 目录 文章目录 前言 1.项目远程仓库配置 2.pom文件引入相关依赖 3.代码破解 二、Excel转PDF 1.代码实现 2.Aspose.License.xml 授权文件 总结 前言 java处理excel转pdf一直没找到什么好用的免费jar包工具,自己手写的难度,恐怕高级程序员花费一年的事件,也…...
如何为服务器生成TLS证书
TLS(Transport Layer Security)证书是确保网络通信安全的重要手段,它通过加密技术保护传输的数据不被窃听和篡改。在服务器上配置TLS证书,可以使用户通过HTTPS协议安全地访问您的网站。本文将详细介绍如何在服务器上生成一个TLS证…...
select、poll、epoll 与 Reactor 模式
在高并发网络编程领域,高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表,以及基于它们实现的 Reactor 模式,为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。 一、I…...
Linux --进程控制
本文从以下五个方面来初步认识进程控制: 目录 进程创建 进程终止 进程等待 进程替换 模拟实现一个微型shell 进程创建 在Linux系统中我们可以在一个进程使用系统调用fork()来创建子进程,创建出来的进程就是子进程,原来的进程为父进程。…...
解读《网络安全法》最新修订,把握网络安全新趋势
《网络安全法》自2017年施行以来,在维护网络空间安全方面发挥了重要作用。但随着网络环境的日益复杂,网络攻击、数据泄露等事件频发,现行法律已难以完全适应新的风险挑战。 2025年3月28日,国家网信办会同相关部门起草了《网络安全…...
【LeetCode】算法详解#6 ---除自身以外数组的乘积
1.题目介绍 给定一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O…...
【把数组变成一棵树】有序数组秒变平衡BST,原来可以这么优雅!
【把数组变成一棵树】有序数组秒变平衡BST,原来可以这么优雅! 🌱 前言:一棵树的浪漫,从数组开始说起 程序员的世界里,数组是最常见的基本结构之一,几乎每种语言、每种算法都少不了它。可你有没有想过,一组看似“线性排列”的有序数组,竟然可以**“长”成一棵平衡的二…...
