政网站首页怎么做试/网络营销的主要方式
🎏:你只管努力,剩下的交给时间
🏠 :小破站
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…...

PHP语言、B/S手术麻醉临床信息管理系统源码
手术麻醉临床信息管理系统有着完善的临床业务功能,能够涵盖整个围术期的工作,能够采集、汇总、存储、处理、展现所有的临床诊疗资料。通过该系统的实施,能够规范麻醉科的工作流程,实现麻醉手术过程的信息数字化,自动生…...

Win11安装网络打印机
https://support.microsoft.com/zh-cn/windows/%E5%9C%A8-windows-%E4%B8%AD%E5%AE%89%E8%A3%85%E6%89%93%E5%8D%B0%E6%9C%BA-cc0724cf-793e-3542-d1ff-727e4978638b...

逆向学习记录(3)工具介绍jadx、gda和jeb
1、jadx 下载地址如下,目前最新版本为v1.4.7,改成想要下载的版本号就能下载对应的版本。 https://github.com/skylot/jadx/releases/tag/v1.4.7 下载后解压,进入对应路径的bin文件夹内,运行jadx-gui.bat。 2、gda 下载地址和gi…...

C#,数值计算——偏微分方程,Mglin的计算方法与源程序
1 文本格式 using System; using System.Collections.Generic; namespace Legalsoft.Truffer { public class Mglin { private int n { get; set; } private int ng { get; set; } private double[,] uj1 { get; set; } private Lis…...

一机服务万人,拓世法宝AI智能商业数字人一体机,解锁文旅新表达
在人工智能的强劲推动下,人们走进了一个令人振奋的数字化时代。如何让文化传承与现代科技完美融合,成为一个十分有趣的议题,当AI技术结合文旅生活,便悄然开启了一种全新的旅游服务模式——AI数字文旅。 在我国国家博物馆、文旅大…...

【源码解析】聊聊SpringBean是如何初始化和创建
我们知道通过类进行修复不同的属性,比如单例、原型等,而具体的流程是怎么样的呢,这一篇我们开始从源码的视角分析以下。 刷新方法 在刷新容器中有一个方法,其实就是 Bean创建的过程。 finishBeanFactoryInitialization(beanFact…...

【0基础学Java第六课】-- 数组的定义与使用
6 数组的定义与使用 6.1 什么是数组6.2 数组的创建及初始化6.2.1 数组的创建:6.2.2 数组的初始化 6.3 数组的使用6.3.1 数组中元素的访问6.3.2 Java中JVM当中的内存划分6.3.3 遍历数组 6.4 数组是引用类型6.4.1 初始JVM的内存分布6.4.2 基本类型变量与引用类型变量的…...

后台项目Gradle打包jar,不包含依赖jar并放到外部路径
# 1.Gradle打包jar # 2.依赖jar包外放到其他目录 # 3.保留引用关系 # 4.去掉引入的缓存build.gradle// 需要放到dependencies下面 // 傻逼问题 1 这个jar打包还得主动开 jar.enabled true // 1.清除上一次的lib目录 task clearJar(type: Delete) {delete "$buildDir\\lib…...

NSSCTF web刷题记录4
文章目录 [NSSRound#4 SWPU]1zweb(revenge)[强网杯 2019]高明的黑客[BJDCTF 2020]Cookie is so subtle![MoeCTF 2021]fake game[第五空间 2021]PNG图片转换器[ASIS 2019]Unicorn shop[justCTF 2020]gofs[UUCTF 2022 新生赛]phonecode[b01lers 2020]Life On Mars[HZNUCTF 2023 f…...

什么是大模型?一文读懂大模型的基本概念
大模型是指具有大规模参数和复杂计算结构的机器学习模型。本文从大模型的基本概念出发,对大模型领域容易混淆的相关概念进行区分,并就大模型的发展历程、特点和分类、泛化与微调进行了详细解读,供大家在了解大模型基本知识的过程中起到一定参…...