小白日更第三十九天->静态变量什么时候被回收

要想真正的知道这个问题的答案,需要一点jvm的基础。

静态变量什么时候分配

静态变量是在类初始化之前就在方法去的静态域中分配内存空间了。

静态变量在类被卸载的时候销毁。

那什么时候类卸载呢?
在<深入理解Java虚拟机:JVM高级特性与最佳实践(第3版) 周志明>3.2.5回收方法区中有这样几段话,供小伙伴们参考~

有些人认为方法区(如HotSpot虚拟机中的元空间或者永久代) 是没有垃圾收集行为的, 《Java虚
拟机规范》 中提到过可以不要求虚拟机在方法区中实现垃圾收集, 事实上也确实有未实现或未能完整
实现方法区类型卸载的收集器存在(如JDK 11时期的ZGC收集器就不支持类卸载) , 方法区垃圾收集
的“性价比”通常也是比较低的: 在Java堆中, 尤其是在新生代中, 对常规应用进行一次垃圾收集通常
可以回收70%至99%的内存空间, 相比之下, 方法区回收囿于苛刻的判定条件, 其区域垃圾收集的回
收成果往往远低于此。
方法区的垃圾收集主要回收两部分内容: 废弃的常量和不再使用的类型。 回收废弃常量与回收
Java堆中的对象非常类似。 举个常量池中字面量回收的例子, 假如一个字符串“java”曾经进入常量池
中, 但是当前系统又没有任何一个字符串对象的值是“java”, 换句话说, 已经没有任何字符串对象引用
常量池中的“java”常量, 且虚拟机中也没有其他地方引用这个字面量。 如果在这时发生内存回收, 而且
垃圾收集器判断确有必要的话, 这个“java”常量就将会被系统清理出常量池。 常量池中其他类(接
口) 、 方法、 字段的符号引用也与此类似。
判定一个常量是否“废弃”还是相对简单, 而要判定一个类型是否属于“不再被使用的类”的条件就
比较苛刻了。 需要同时满足下面三个条件:
·该类所有的实例都已经被回收, 也就是Java堆中不存在该类及其任何派生子类的实例。
·加载该类的类加载器已经被回收, 这个条件除非是经过精心设计的可替换类加载器的场景, 如
OSGi、 JSP的重加载等, 否则通常是很难达成的。
·该类对应的java.lang.Class对象没有在任何地方被引用, 无法在任何地方通过反射访问该类的方
法。
Java虚拟机被允许对满足上述三个条件的无用类进行回收, 这里说的仅仅是“被允许”, 而并不是
和对象一样, 没有引用了就必然会回收。 关于是否要对类型进行回收, HotSpot虚拟机提供了-
Xnoclassgc参数进行控制, 还可以使用-verbose: class以及-XX: +TraceClass-Loading、 -XX:
+TraceClassUnLoading查看类加载和卸载信息, 其中-verbose: class和-XX: +TraceClassLoading可以在
Product版的虚拟机中使用, -XX: +TraceClassUnLoading参数需要FastDebug版[1]的虚拟机支持。
在大量使用反射、 动态代理、 CGLib等字节码框架, 动态生成JSP以及OSGi这类频繁自定义类加载
器的场景中, 通常都需要Java虚拟机具备类型卸载的能力, 以保证不会对方法区造成过大的内存压
力。

总结:
·该类所有的实例都已经被回收, 也就是Java堆中不存在该类及其任何派生子类的实例。
·加载该类的类加载器已经被回收, 这个条件除非是经过精心设计的可替换类加载器的场景, 如OSGi、 JSP的重加载等, 否则通常是很难达成的。
·该类对应的java.lang.Class对象没有在任何地方被引用, 无法在任何地方通过反射访问该类的方法。

以上就是我的理解,以及书中的内容。(好像都是书中的内容哈哈)
在这里插入图片描述

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是Test.java文件的代码: ``` public class Test { public static void main(String[] args) { Leaf leaf1 = new Leaf(); Leaf leaf2 = new Leaf(); } } class Root { static { System.out.println("Root的静态代码块"); } { System.out.println("Root的实例代码块"); } public Root() { System.out.println("Root的构造函数"); } } class Mid extends Root { static { System.out.println("Mid的静态代码块"); } { System.out.println("Mid的实例代码块"); } public Mid() { this("调用重载的构造器"); System.out.println("Mid的无参构造函数"); } public Mid(String s) { System.out.println("Mid的带参构造函数:" + s); } } class Leaf extends Mid { static { System.out.println("Leaf的静态代码块"); } { System.out.println("Leaf的实例代码块"); } public Leaf() { super("调用父类指定的构造器"); System.out.println("Leaf的无参构造函数"); } } ``` 输出结果为: ``` Root的静态代码块 Mid的静态代码块 Leaf的静态代码块 Root的实例代码块 Root的构造函数 Mid的实例代码块 Mid的带参构造函数:调用重载的构造器 Mid的无参构造函数 Leaf的实例代码块 Leaf的带参构造函数:调用父类指定的构造器 Leaf的无参构造函数 Root的实例代码块 Root的构造函数 Mid的实例代码块 Mid的带参构造函数:调用重载的构造器 Mid的无参构造函数 Leaf的实例代码块 Leaf的带参构造函数:调用父类指定的构造器 Leaf的无参构造函数 ``` 可以看到,初始化代码块和构造函数的执行顺序符合题目中所给的加载顺序。同时,Leaf类使用super显示调用其父类指定的构造器,而Mid类使用this调用重载的构造器,验证了题目中的要求。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值