• JVM 类的卸载


    1、当某个类被加载,连接和初始化后,它的生命周期就开始了。当代表这个类的Class对象不再被引用,即不可触及时,Class对象就会结束生命周期,这个类在方法区内的数据也会被卸载,从而结束这个类的生命周期。

    2、一个类何时结束生命周期,取决于代表它的Class对象何时结束生命周期。

    3、由Java虚拟机自带的类加载器所加载的类,在虚拟机的生命周期中,始终不会被卸载。前面已经介绍过,Java虚拟机自带的类加载器包括根加载器、扩展类加载器和系统加载器。Java虚拟机会始终引用这些类加载器,而这些类加载器则会始终引用它们所加载的Class对象,因此这些Class对象始终是可触及的。

    4、由用户自定义的类加载器所加载的类是可以被卸载的。

    5、运行以上程序时,Sample类由loader1加载。在类加载器的内部实现中,用一个Java集合来存放所加载类的引用。另一方面,一个Class对象总是会引用它类加载器,调用Class对象的getClassLoader()方法,就会获得它的类加载器。由此可见,代表Sample类的Class实例与loader1之间为双向关联关系。

       一个类的实例总是引用代表这个类的Class对象。在Object类中定义了getClass()方法,这个方法返回代表对象所属的Class对象的引用。此外,所有的Java类都有一个静态属性class,它引用代表这个类的Class对象。

    6、类卸载的例子

    public class MyTest161 extends  ClassLoader{
    
        private String className;
    
        //目录
         private String path;
    
        private final String fileExtension = ".class";
    
        public MyTest161(String classLoadName){
            super(); //将系统类加载器当做该类加载器的父加载器
            this.className = classLoadName;
        }
    
        public MyTest161(ClassLoader parent, String classLoadName){
            super(parent); //显示指定该类加载器的父加载器器
            this.className = classLoadName;
        }
    
        public void setPath(String path) {
            this.path = path;
        }
    
        @Override
        public String toString() {
            return "[" + this.className + "]";
        }
    
        @Override
        protected Class<?> findClass(String clasName) throws ClassNotFoundException {
            System.out.println("findClass invoked:" + clasName);
            System.out.println("class loader name: " + this.className);
            byte[] data = this.loadClassData(clasName);
            return  this.defineClass(clasName,data, 0, data.length);
        }
    
        private byte[] loadClassData(String className){
            InputStream is = null;
            byte[] data = null;
            ByteArrayOutputStream baos = null;
    
            try{
                className = className.replace(".","//");
                //System.out.println("className:" +this.className);
                is = new FileInputStream(new File(this.path + className + this.fileExtension));
                baos = new ByteArrayOutputStream();
                int ch = 0;
                while ( -1 != (ch = is.read())){
                    baos.write(ch);
                }
                data = baos.toByteArray();
    
            }catch (Exception ex){
                ex.printStackTrace();
            }finally {
                try {
                    is.close();
                    baos.close();
                }catch (Exception ex){
                    ex.printStackTrace();
                }
            }
    
            return  data;
    
        }
    
        public static void main(String[] args)  throws Exception{
            MyTest161 loader1 = new MyTest161("loader1");
            loader1.setPath("D:/temp/");
            Class<?> clazz = loader1.loadClass("com.example.jvm.classloader.MyTest1"); //
            System.out.println("class:" + clazz.hashCode());
            Object object = clazz.newInstance();
            System.out.println(object);
            loader1 = null;
            clazz = null;
            object = null;
            System.gc(); // 垃圾回收。实际开发不会用
            System.out.println();
    
            loader1 = new MyTest161("loader1");
            loader1.setPath("D:/temp/");
            clazz = loader1.loadClass("com.example.jvm.classloader.MyTest1"); //
            System.out.println("class:" + clazz.hashCode());
            object = clazz.newInstance();
            System.out.println(object);
            System.out.println();
    
    
    
        }
    
    
    
    }
    

      设置JVM参数 -XX:+TraceClassUnloading

    运行结果:

    findClass invoked:com.example.jvm.classloader.MyTest1
    class loader name: loader1
    class:21685669
    com.example.jvm.classloader.MyTest1@7f31245a
    
    findClass invoked:com.example.jvm.classloader.MyTest1
    class loader name: loader1
    class:1173230247
    [Unloading class com.example.jvm.classloader.MyTest1 0x0000000100061028]
    com.example.jvm.classloader.MyTest1@330bedb4
    

      

     7、使用JVisualVM查看类的卸载

     增加睡眠1分钟(JVisualVM的使用参考JVisualVM监控本地Java进程

    然后使用JVisualVM观察,可以发现已经卸载类的总数为1

  • 相关阅读:
    [BZOJ]2132: 圈地计划 最小割
    从最近MySQL的优化工作想到的
    Linux基本操作 9----- 认识与学习bash
    多路径配置vlome group共享存储,VG的更新。
    两位数乘法的速算方法(一)
    请对他有足够的重视——设计!
    ASP.NET中配置应用程序
    flex开发小技巧集锦
    刚制作完的SAP Sybase ASE15.7 [Sybase Central] 客户端
    Static 关键字的 5 种用法,你会几种?
  • 原文地址:https://www.cnblogs.com/linlf03/p/11020251.html
Copyright © 2020-2023  润新知