最新动态
jvm+反编译深入了解final,static关键字的作用
2024-11-24 03:17  浏览:53
  • 变量
  • 方法
  • 内部类
  • 成员变量
  • 代码块
  • 方法

保证数据的一致性。

jvm+反编译深入了解final,static关键字的作用

被 final 修饰的类被称为最终的类 / 不可变类。例如 String 类,设计者认为 String 已经完美了,不想因为 String 被随意继承、重写方法而导致的错误。

光使用 final 修饰一个类还达不到不可变类的标准,还要保证:1. 成员变量私有化,并被 final 修饰。2.不向外提供修改成员变量的方法 3.当类中含有可变的属性时(数组、对象,该属性对应的 getter 方法不直接返回对象的引用,而是新创建一个内容相同的对象并返回其引用。java 中 String 以及 Integer、Double 等包装类都是不可变类。 String的不可变性

这里的变量分为两种 1.成员变量 2.局部变量

3.2.1 成员变量

final 修饰的成员变量必须被初始化。有关成员变量初始化的时机,稍后会讲到:跳转

当被 final 修饰的成员变量为基本数据类型时,成员变量的值不可变,即常量

当成员变量为引用数据类型时(数组、对象,成员变量存储的引用不可变,但引用指向的值可以变。(final 修饰引用类型没什么意义

举个例子

代码一

 

BrainStorm

我们知道当引用类型变量被 final 修饰后,必须完成初始化,且不能二次赋值,也就是引用不可变。但当垃圾收集时,对象被移动是常有的事。所以说 final 修饰的对象不会被垃圾回收吗

3.2.2局部变量

以下代码版本 jdk7

代码二
 

jdk8前,当 testIn() 方法想要访问外部类 test() 方法中的局部变量 i 时,i 需要被 final 修饰。

为什么呢?有两种说法。第一种说法是匿名内部类的对象与局部变量的生命周期不同。第二种说法是防止后续代码修改局部变量。

探究1、先来探究第一种说法

同意第一种说法的人认为:“匿名内部类对象的生命周期大于局部变量的声明周期,被 final 修饰的局部变量才能被匿名内部类对象访问,以防当匿名内部类对象需要访问这个局部变量的时候出现访问不到而导致错误。”

先来看第一句话

我们知道,JVM 层面上,方法的调用对应的是虚拟机栈中栈帧的压栈和弹栈。局部变量属于栈帧中的局部变量表,局部变量随着方法的调用而生存,也随着方法的结束(方法运行到 return 字节码指令时结束)而消亡。

匿名内部类对象当没有引用指向它时,才会被 GC 回收。

所以匿名内部类对象生命周期看似与局部变量的生命周期不同。

探究2、是否存在生命周期不同的情况

 

以上代码中 test() 创建了匿名内部类并返回,由 main 方法调用匿名内部类对象的 testIn() 方法。当 testIn() 方法运行时,test() 方法以及结束(已出栈),局部变量已经死亡。而匿名内部类对象还存活,并且 testIn() 需要访问局部变量 test。

这种情况下,匿名内部类对象的生命周期确实与局部变量的生命周期不同。

再来看第二句话

第一种说法认为,为了防止生命周期不同导致异常,需要使用 final 修饰被匿名内部类对象访问的局部变量" "。让我们回忆一下,final 的作用到底是什么?是保证数据的一致性。我们很难联想到 final 与访问权限有什么关系。我们暂且搁置这个问题,先来看看匿名内部类对象是如何访问局部变量的。

探究3、我们再来看看匿名内部类对象是如何访问局部变量的

由于 java 的封装性,类的外部无法访问类内部的私有属性,更不可能访问方法的局部变量。那么匿名内部类是如何访问方法的局部变量呢?其实是通过值传递的方式,将变量的值传递给匿名内部类对象。那么是如何传递的呢

用语言描述不好理解,我们来看 FinalTest2 类(代码二)的编译结果,生成了两个 Class 文件

​ 1.FinalTest2.class

在这里插入图片描述

​ 2.FinalTest2 表示动态生成

在这里插入图片描述

局部变量 i 的值传递到 testIn() 方法中。

而当局部变量不是基本数据类型时,匿名内部类对象访问局部变量的方式有所区别。

代码三

 

在这里插入图片描述

final 修饰的 obj 作为参数传入 FinalTest2$1 的有参构造。FinalTest2$1 中会有一个成员变量来保存 test的引用。保证 FinalTest2$1 的成员变量 val $test 与 test() 方法中的局部变量 test 指向栈中同一个地址。也就是将局部变量拷贝一份,传递给匿名内部类的成员变量。

回看探究2

现在我们明白了匿名内部类是如何访问局部变量的,让我们来解决探究二搁置的问题。

当局部变量是引用类型时,匿名内部类对象访问的是局部变量的拷贝。如果局部变量没有被 final 修饰,局部变量完全有可能在被拷贝以后,指向新的引用。而此时匿名内部类对象是无从知晓,因为拿到的只是一份指向同一个引用的拷贝。并且因为拿到的只是一份拷贝,所以局部变量在方法结束时回收就回收了,并不影响匿名内部类对象访问。

当局部变量是基本数据类型时,如果局部变量没有被 final 修饰,值也可以被随意更改。

那匿名内部类对象就不干了:“由于 java 的封装性,我不能直接访问局部变量。所以我废半天劲拷贝局部变量,就是为了访问它。结果你告诉我,这是个二手货!我裤子都脱了,你给我看这个?”。

所以 java 为了防止匿名内部类对象访问到真的局部变量,强制让 final 修饰被访问的局部变量,不让局部变量的值发生改变。这也就证实了,final 的语义只是一句简单的:保证数据的一致性。

jdk8 以后,不需要我们显式的使用 final 修饰局部变量 ,javac 编译器会隐式的帮我们加上。(Effectively final)

探究4、再来说说第二种说法

代码三

第二种说法是防止后续代码修改局部变量。综合上面的探究,我认为这种说法是正确的。

BrainStorm

有人认为,匿名内部类对象没有名字,那么也就没有构造方法。所以匿名内部类对象不是通过有参构造,将拷贝而来的局部变量赋值给自己的成员变量的,而是通过初始化代码块赋值的。我们通过反编译查看字节码指令:

 

通过反编译我们一目了然,invokespecial 字节码指令调用 FinalTest2$1 的有参构造。

当方法被 final 修饰时,不能被重写 Overrite,但没有不允许被重载 Overload。

被 private 、static 修饰的方法默认是 final 的。

内联函数,提高效率。(我不太了解,以后会来补上!)

宏替换有两点要求

  1. 基本数据类型,String
  2. 在定义时就完成初始化
  3. 编译期间能确切知晓常量的值

例如以下这个例子

 

编译优化为

 

将被 static 修饰的属性、方法、内部类与类进行关联,随着类的信息进入方法区。被该类的所有实例共享。

跳转

static 关键字可以用于修饰成员变量,不能用于修饰局部变量。被 static 修饰的成员变量属于类,该类的所有实例共享同一个静态变量。

被 static 修饰的代码块,在类加载阶段中的初始化阶段运行,且只运行一次。

javac 编译后,修饰符为 static final 的变量会存在于类文件中的 ConstantValue 属性中。 在类加载的准备阶段会为类变量分配内存并赋零值 当被 static final 修饰的变量数据类型为基本类型或者 String 类型时,在类加载的准备阶段时就会为变量赋值(不是零值)

​ 类构造方法 < clinit >(),它会收集类中的静态成员变量及赋值语句(显式初始化)、静态代码块: static {…} ,并在类初始化时,运行该方法。

​ 实例构造方法 < init >(),它会收集类中的实例成员变量(显式初始化)、实例代码块: { … }、以及无参构造方法、以及父类的实例初始化方法,并在创建对象时运行该方法。没有显式初始化的实例成员变量,在何时初始化呢?当对象被创建,JVM 为对象分配完内存后,会为除对象头外的内存空间初始化(赋零值)

​ 需要注意,当成员变量被 final 修饰时:静态成员变量必须在类构造方法完成前完成初始化,实例成员变量必须在实例构造方法完成前完成初始化。

try:判断一下下面这个类的 < clinit >(),< init >()内容会是什么

 

result:反编译结果

  1. ()方法内容调用父类实例初始化方法,初始化 i3(i4 没有显式初始化,init 不收集),调用实例代码块,调用无参构造。
 

细节:< init >() 方法没有传入参数,但为什么 args_size=1呢?this 关键字。Java 中实例方法会默认接收 this 关键字(方法的调用者,而静态方法不会接收,因为静态方法是属于类的,只能由类调用。

  1. < clinit >()方法内容:初始化i1 (i2 没有显式初始化,clinit 不收集),调用静态代码块
 

BrainStorm

静态成员是如何被所有实例共享的呢?原理是什么

静态成员是属于类的,位于方法区(逻辑区域)中(真实物理区域存在于元空间中)。每个类在堆中都有一个对应的 class 对象,作为方法区中类的信息的入口。

当我们获取静态成员时,实际上是调用了 getstatic 字节码指令,通过类对应的 class 对象去获取静态变量的值。同样,修改静态成员的值时,实际上调用了 putstatic 字节码指令。

静态内部类方式单例

利用1.类只会被加载一次 2.静态成员被所有实例共享 两个特点我们可以通过静态内部类的方式,实现单例模式

 

static 修饰的方法是属于类的,默认是 final 的,不会被子类继承。

在这里插入图片描述

在静态方法中可以调用实例方法,但在实例方法中不能调用静态方法,这是什么原因呢

静态方法会在类加载阶段中就分配内存,而实例方法是在创建对象时被分配到内存。如果在静态方法中访问实例数据或者调用实例方法,jvm 找不到对应的内存地址,也就不可能解析成功。

被 static final 修饰的变量称为:全局常量。这个常量,全局共享,且值无法被更改。(例如上面的单例模式)

为什么接口中的属性默认为static final

    以上就是本篇文章【jvm+反编译深入了解final,static关键字的作用】的全部内容了,欢迎阅览 ! 文章地址:http://lanlanwork.gawce.com/quote/8829.html 
     行业      资讯      企业新闻      行情      企业黄页      同类资讯      网站地图      返回首页 阁恬下移动站 http://lanlanwork.gawce.com/mobile/ , 查看更多