final关键字_深入分析Java中的final关键字

f1f86b527b5f0c639bdf7f03fb60033b.png

Java中被final修饰的变量与普通变量有何区别?被final修饰的变量不可更改、被final修饰的方法不可重写是怎样做到的?带着疑问我们一点点拨开云雾。

一、final的内存定义及规则

对于final关键字,编译器、处理器从读写两个角度限制了其使用规则:

  • 对于一个类的final修饰的变量,如果在定义是不指定初始值,那么在构造函数中必须进行初始化,在构造函数中进行final域的写入时,随后将构造后的对象引用赋值给另外一个引用变量,它们之间不能进行重排序的发生。
  • 在读一个包含final关键字的对象引用和读这个引用的包含的final修饰的变量时,这两个操作间不能发生重排序。

下面通过一段代码分析一下具体场景:

public 

这里先假设A线程执行finalWriter方法,B线程执行finalRead()方法,通过上述对于final的规则描述我们分析一下finalWriter方法的执行流程:

  1. 构造一个FinalTest对象
  2. 将构造后的对象引用进行赋值

对于final修饰的变量进行赋值操作时的重排序规则如下:

1、Java内存模型禁止将对final关键字修饰的变量进行写操作重排序到构造函数之外。
2、编译器会在写之后,构造函数return之前插入StoreStore屏障,这个屏障确保编译器不会把final变量写操作重排序到构造函数之外。

下面假定一种重排序的场景如下图所示:

1179bab56f41622b6d4f0b2be34c2390.png
final域的写流程

上图的场景情况为变量i为普通变量,在进行赋值时发生了重排序(由于这时候有可能构造函数还未完成),在构造函数结束后,才进行了赋值,线程B读取到的i的值为赋值前的初始值0,而对于final修饰的变量j由于禁止重排序,在构造函数return前需要进行赋值,限定到了构造函数内,读取到的变量j为正确的值。

然后再分析一下执行finalRead()方法的流程:

  1. 读引用变量Obj将其赋值给object变量。
  2. 读引用的普通变量i。
  3. 读引用的final修饰的变量j

对于final修饰的读操作重排序规则:

在一个线程中首次读对象的引用和首次读该对象包含的final修饰的变量,Java内存模型禁止重排序(也就是说在读取一个final修饰的变量前,一定是先获取该变量对应的引用),主要原理就是在读取final修饰的变量前插入LoadLoad屏障。

假设上述情况,线程A正常执行,变量i没有发生重排序的情况,而对于线程B读取变量i和读对象的引用发生了重排序,如下图所示

5010ab7b2f7b84d17ecb63b9f2d47580.png

读对象的普通变量i时处理器发生了重排序,读变量在读对象的引用之前发生,这时候变量还未开始进行赋值,而对于final修饰的变量j来说,由于其遵循重排序规则(读变量首先要读变量对应的对象引用),所以读取的值是正确的。

除了上述两种场景之外,假设在构造函数内使用this关键字将当前对象赋值给成员变量(逸出),如下代码所示:

public 

这时候同样有两个线程,一个线程执行finalWriter,另外一个执行finaRead,也有可能会出现被final修饰的变量j没有进行赋值的情况。

参考《Java并发编程的艺术》

0452adc9aff4b13689ef5a76b170b790.png
扫码关注“聊点源码”获取更多资讯

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/561135.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

嵌入式和fpga哪个好前景_Java 和 go 哪个就业前景好一点?面向金钱编程,这个回答太现实...

有个朋友问我:ava 和 go 哪个就业前景好一点?现在纠结中,差不多一年半就要毕业了。我也问了几个朋友程序员朋友:朋友A: 可以参考 Boss 直聘,面向金钱编程,我身边的 Java 都是 15k 左右,反正干就…

php解决mysql主从同步_Mysql读写分离,主从同步实现

随着用户量的增多,数据库操作往往会成为一个系统的瓶颈所在,因此我们可以通过实现数据库的读写分离来提高系统的性能。通过设置主从数据库实现读写分离,主库负责“写”操作,从库负责“读”操作,根据压力情况&#xff0…

python创建数组的方法_numpy创建array的方法汇总

创建numpy.array,是使用numpy这个核武器的基础,本文尽量汇总常用创建numpy.array的方法。array函数>>> import numpy as np>>> a np.array([1,2,3,4,5])>>> aarray([1, 2, 3, 4, 5])>>> a.shape(5,)>>> a…

mysql_install_db is deprecated_MySQL5.7源码安装问题汇总

编译安装mysql5.7版本,想试用一下新的版本特性,发现跟之前的5.6版本编译有了一些变化,总结一下避免以后继续入坑。5.6安装方式cmake版本5.7编译cmake要求版本最低为2.8,当前为2.6,所以需要升级cmake版本。信息如下shel…

验证码图片显示不出来怎么办_pr 的蒙版不显示了怎么办?

小白自学pr一路会遇到很多坑,我也是在一个个坑里爬过的。画好的蒙版不显示了,这只能算是一个小坑。解决方法很简单,只需要记住一条:在pr里想要调整或者显示什么东西,首先必须要选中它。举例说明:我给小猫的…

gb2312编码表_汉字编码输入系统模型(一)

通过前面介绍的通信系统模型来分析汉字编码输入系统,将汉字编码输入系统的特殊性整合到通信系统模型中,从而建立起一种基于信息论的汉字编码输入系统模型(参见图3.2),以便指导我们的汉字编码输入实践,设计和…

python dict hash_【python-dict】dict的使用及实现原理

以下内容是针对:python源码剖析中的第五章——python中Dict对象 的读书笔记(针对书中讲到的内容进行了自己的整理,并且针对部分内容根据自己的需求进行了扩展)一、Dict的用法Dict的对象在使用到了所谓的关联关系的时候,就是通过key-value的形…

python参考文献_[zotero/python]库中参考文献条目删除后,清除残留PDF的脚本

更新:使用 滏阳河边捉蚯蚓 https://zhuanlan.zhihu.com/p/41297136上获取系统PDF文件和zotero.sqlite文件的代码,在此感谢!在zotero的library中删除参考文献条目后,有时PDF不会同步删除,尤甚是安装了zotfile插件后&…

java中sql语句怎么把开始和结束时间作为参数写sql查询_JDBC数据库连接怎么操作?...

之前一直听说过JDBC,但从来不知道它是何物的小伙伴们看过来啦!一、概述JDBC(Java Data Base Connectivity,java数据库连接)是一种用于执行SQL语句的Java API,可以为多种关系数据库提供统一访问,它由一组用Java语言编写的类和接口组…

spring boot整合shiro继承redis_spring-boot-plus集成Shiro+JWT权限管理

SpringBootShiroJWT权限管理ShiroApache Shiro是一个强大且易用的Java安全框架,执行身份验证、授权、密码和会话管理。使用Shiro的易于理解的API,您可以快速、轻松地获得任何应用程序,从最小的移动应用程序到最大的网络和企业应用程序。三个核心组件:Subject, Secur…

java 整数变负数_一文帮你读懂Java整数的存储原理

前言大家应该都知道,整数包括负数,零,和正数。在Java中,基本类型中byte(8位)、short(16位)、int(32位)、long(64位)属于整数,并且没有无符号数,均是有符号的。对于计算机来说,它只认识二进制&am…

server2019 sqlcmd命令安装_Ubuntu20.04LTS安装MS sql-server2019的方法

在本文中,我将向您展示如何在Ubuntu 20.04 Focal Fossa上安装Microsoft SQL Server。小广告:喜欢网络技术的朋友可以加W信:xfiles_sky一起学习进步。步骤一:更新ubuntu源sudo apt update && sudo apt upgrade步骤二&#…

servlet里面为什么有时候覆_为什么新来的经理强烈推荐?前后端分离知识,学到了...

引言前后端分离这个词相信大家都听过,不知道大家是怎么理解的呢。前阵子看项目的时候,有一段实现硬是没看懂,下面来给大家说一下一段愚蠢的经历哈。(我没正正式式写过前端,所以如果文章有错的地方希望可以在评论区友善交流~)一、交…

lamp 安装 mysql_linux lamp之离线安装mysql

Linux下自己创建目录/tigger上传mysql压缩包到此目录[rootlocalhost tigger]# tar zxvf mysql.tar.gz[rootlocalhost tigger]# cp -r mysql /usr/local/mysql/usr/local/mysql 为安装目录cd /usr/local/mysql[rootlocalhost mysql]# groupadd mysql[rootlocalhost mysql]# user…

microstation添加txt文件_C开发实战-文件操作

文件概述文件几乎无处不在,主要分为磁盘文件和设备文件,典型的磁盘文件有文本文件和二进制文件,磁盘文件存储在外部存储介质(例如磁盘,硬盘,U盘等等)需要加载到内存中才能使用。无论是文本文件还是二进制文件在计算机内…

怎样配置mysql数据源_mysql怎样配置ODBC数据源

选中 sqlserver选择默认连接的数据库 7、配置完成,可以测试连接 扩展资料: spring中配置数据源的几种常见方式:工具/原料 事先配置相应的环境mysql(mysql安装程序)mysql-connector-odbc-3.51.20-win32.exe(mysql数据源dobc安装程序) 步骤/方法…

python开发框架大全_最受欢迎 Top 12 Python 开源框架,你都用过吗?

作者 | 学Python的阿勇责编 | 夕颜出品 | CSDN博客今天给大家带来了12个在GitHub等开源网站中最受欢迎的Python开源框架。如果你正在学习python,那么这12个开源框架,千万别错过,这些框架包括事件I/O,OLAP,Web开发&…

python中的zip是什么意思_Python的zip()函数是什么

zip() 函数可以把两个列表“压缩”成一个 zip 对象(可迭代对象),这样就可以使用一个循环并行遍历两个列表。为了测试 zip() 函数的功能,我们可以先在交互式解释器中“试验”一下该函数的功能。>>> a [a,b,c]>>> b [1, 2, 3]>>…

python 子线程返回值_python-从线程返回值

python-从线程返回值我如何获得一个线程以将元组或我选择的任何值返回给Python中的父级?12个解决方案59 votes我建议您在启动线程之前实例化Queue.Queue,并将其作为线程的args之一传递:在线程完成之前,它.puts将其结果作为参数接收…

python实现火车票查询_python实现12306火车票查询器

12306火车票购票软件大家都用过,怎么用Python写一个命令行的火车票查看器,要求在命令行敲一行命令来获得你想要的火车票信息,下面通过本文学习吧。Python火车票查询器接口设置先给这个小应用起个名字吧,既然及查询票务信息&#x…