• 类的加载机制(五)


    类加载的过程和原理,以及双亲委派机制都已经讲解完成,那么我们今天讲解类加载的最后一节,那么就是类的卸载。

    我们知道,当一个类被加载、连接和初始化之后,他的生命周期就开始了,当该类的class对象不再被引用之后,该类的生命周期也就结束了,之后,该类会被类加载器卸载!
    我们来看以下代码:

    复制代码
    public class UninstallTest {
    
    
    
        public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException {
    
           MyClassLoader classLoader1 = new MyClassLoader("classLoader1");
    
           classLoader1.setClassPath("d:/jvmTest/server/");
    
           Class<?> clazz = classLoader1.loadClass("TestCase1");
    
           @SuppressWarnings("unused")
    
           Object obj = clazz.newInstance();
    
           System.out.println("1:"+clazz.hashCode());
    
           obj=null;
    
           System.out.println("2:"+clazz.hashCode());
    
           classLoader1 = null;
    
           System.out.println("3:"+clazz.hashCode());
    
           clazz = null;
    
           System.out.println("===========");
    
           classLoader1 = new MyClassLoader("classLoader1");
    
           classLoader1.setClassPath("d:/jvmTest/server/");
    
           clazz = classLoader1.loadClass("TestCase1");
    
           System.out.println("4:"+clazz.hashCode());
    
        }
    
    }
    复制代码

    以下代码先通过自定义类加载器classloader1加载类Testcase1,然后通过反射获取对应的obj对象,然后逐个将对应的对象设置为null以便于销毁对象,执行结果是是什么样子的呢?

    我们看到,前三次打印对象的hashCode是一致的,说明是一个对象,但是最后一次却反生了变化,说明对象的地址变化了,也就是之前被卸载了(因为我们知道,类的class对象在整个类加载过程中只会被加载一次,只有一个实例对象,除非类被卸载了),然后又被加载了一次(加载了2次)。
    我们再来看这几个类的关系,我们把它画成图,如下:

    栈空间变量指向堆控件的数据,已经没什么争论,之前(上几篇文章)已经讨论过,loader1对象可以通过findClass方法拿到对应的TestCase1类,因此有一个指针指向TestCase1的Class实例,而TestCase1可以通过getClassLoader的方法拿到对应的类加载器,因此有一个指针指向MyClassloader的对象,TestCase1的实例obj通过getClass方法可以拿到对应的TestCase1实例。我们这样可以看到,MyClassloader对象有2个引用指针,TestCase1有3个引用指针,当cloder1为null,objClass为null,obj为null的时候,系统还剩下MyClassloader和TestCase1的一个相互引用。

    我们知道,只要class引用不在有,那么这个类就可以回收了,指向方法区的就不看了,但是MyClassloader和TestCase1每个还有一个引用么人这两个其实已经没有其他引用了,但是从程序结果我们可以看到,类已经被卸载了,说明JVM已经对这种双向引用也做了卸载和回收操作

  • 相关阅读:
    【BZOJ2655】—calc(拉格朗日插值+生成函数+dp)
    【BZOJ4559】【JLOI2016】—成绩比较(拉格朗日插值+dp)
    【BZOJ5339】【洛谷P4593】【TJOI2018】—教科书般的亵渎(拉格朗日插值)
    【LOJ2542】【PKUWC2018】—随机游走(Min-Max容斥+树形dp)
    【LOJ#121】—「离线可过」动态图连通性(线段树分治)
    【BZOJ4867】【洛谷P5210】【ZJOI2017】—线段树(括号序列+树链剖分)
    【洛谷P4191】【CTSC2010】—性能优化(混合基FFT)
    【BZOJ3112】【ZJOI2013】—防守战线(线性规划+对偶)
    python 面向对象介绍
    lsof 命令
  • 原文地址:https://www.cnblogs.com/huxipeng/p/8686916.html
Copyright © 2020-2023  润新知