深入理解ClassLoader(五)—类的卸载

原文地址:http://yhjhappy234.blog.163.com/blog/static/316328322011101413730764/?suggestedreading&wumii

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

我们来看以下代码:

package com.yhj.jvm.classloader.uninstall;< xmlnamespace prefix ="o" ns ="urn:schemas-microsoft-com:office:office" />

import com.yhj.jvm.classloader.user_defined.MyClassLoader;

/**

 * @Described:类的卸载测试

 * @author YHJ create at 2011-10-17 下午10:15:52

 * @FileNmae com.yhj.jvm.classloader.uninstall.UninstallTest.java

 */

public class UninstallTest {

    /**

     * @param args

     * @throws ClassNotFoundException

     * @throws IllegalAccessException

     * @throws InstantiationException

     * @Author YHJ create at 2011-10-17 下午10:15:40

     */

    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已经对这种双向引用也做了卸载和回收操作,但是按照我们之前的理论,只有当引用计数器为0的时候才会被回收,这又如何理解呢?这个我们会在下面的文章,深入理解JVM—垃圾回收机制做详细介绍,敬请期待哦!

时间: 2024-10-25 19:55:15

深入理解ClassLoader(五)—类的卸载的相关文章

深入理解ClassLoader(四)—类的父委托加载机制

上几次我们介绍到了JVM内部的几个类加载器,我们来重新画一下这个图,再来看一下他们之间的关系. JVM的ClassLoader采用的是树形结构,除了BootstrapClassLoader以外?每个ClassLoader都会有一个parentClassLoader,用户自定义的ClassLoader默认的parentClassLoader是SystemClassLoader,当然你可以自己指定需要用哪一个ClassLoader的实例,我们来看他的API 默认的无参构造方法使用的是SystemCl

【深入理解Java虚拟机 】类加载器的命名空间以及类的卸载

类加载器的命名空间 每个类加载器又有一个命名空间,由其以及其父加载器组成 类加载器的命名空间的作用和影响 每个类加载器又有一个命名空间,由其以及其父加载器组成 在每个类加载器自己的命名空间中不能出现相同类名的类 (此处值得是类的全名,包含包名) 在不同的类命名空间中,可能会出现多个相同的类名的类 如下面的代码示例中, 首先定义一个类加载器 MyClassLoader static class MyClassLoader extends ClassLoader { private String c

深入理解ClassLoader工作机制(jdk1.8)

ClassLoader 顾名思义就是类加载器,ClassLoader 作用: 负责将 Class 加载到 JVM 中    审查每个类由谁加载(父优先的等级加载机制)    将 Class 字节码重新解析成 JVM 统一要求的对象格式 类加载时机与过程 类从被加载到虚拟机内存中开始,直到卸载出内存为止,它的整个生命周期包括了:加载.验证.准备.解析.初始化.使用和卸载这7个阶段.其中,验证.准备和解析这三个部分统称为连接(linking).这里写图片描述 其中,加载.验证.准备.初始化和卸载这五

五类常见算法小记 (递归与分治,动态规划,贪心,回溯,分支界限法)

近日复习了一些算法知识,小记于此 递归与分治法 直接或间接地调用自身的算法称为递归算法. 递归是算法设计与分析中常用的一种技术,描述简单且易于理解. 分治法的设计思想是将一个规模为n难以解决的问题分解为k个规模较小的子问题,这些子问题相互独立且与原问题相同. 递归地解这些子问题,然后将各子问题的解合并得到原问题的解. 典型例子:Fibonacci数列,阶乘,Hanoi塔:二分法搜索.快速排序.合并排序. 动态规划法 动态规划过程是:根据当前(阶段)状态,采取相应的决策,引起状态的转移.如下图,一

iOS runtime探究(一): 从runtime开始理解面向对象的类到面向过程的结构体

你要知道的runtime都在这里 转载请注明出处 http://blog.csdn.net/u014205968/article/details/67639205 本文主要讲解runtime相关知识,从原理到实践,由于包含内容过多分为以下五篇文章详细讲解,可自行选择需要了解的方向: 从runtime开始: 理解面向对象的类到面向过程的结构体 从runtime开始: 深入理解OC消息转发机制 从runtime开始: 理解OC的属性property 从runtime开始: 实践Category添加属

(转)理解classloader

ClassLoader翻译过来就是类加载器,普通的Java开发者其实用到的不多,但对于某些框架开发者来说却非常常见.理解ClassLoader的加载机制,也有利于我们编写出更高效的代码.ClassLoader的具体作用就是将class文件加载到jvm虚拟机中去,程序就可以正确运行了.但是,jvm启动的时候,并不会一次性加载所有的class文件,而是根据需要去动态加载.想想也是的,一次性加载那么多jar包那么多class,那内存不崩溃.本文的目的也是学习ClassLoader这种加载机制. 备注:

Java反射理解(五)-- 方法反射的基本操作

Java反射理解(五)-- 方法反射的基本操作 方法的反射 1. 如何获取某个方法 方法的名称和方法的参数列表才能唯一决定某个方法 2. 方法反射的操作 method.invoke(对象,参数列表) 举例 具体操作请看下面举例: import java.lang.reflect.Method; class A{ public void print(){ System.out.println("helloworld"); } public void print(int a,int b){

最常用的五类CSS选择器

一些新手朋友对选择器一知半解,不知道在什么情况下运用什么样的选择器,这是一个比较头疼的问题,针对新手朋友,对CSS选择器作一些简单的说明,希望能对大家的学习工作有一定的帮助,更多的CSS知识请参考Webjx.com的其他文章. 准确而简洁的运用CSS选择器会达到非常好的效果.我们不必通篇给每一个元素定义类(class)或ID,通过合适的组织,可以用最简单的方法实现同样的效果.在实际工作中,最常用的选择器有以下五类: 一.标签选择器: 顾名思议,标签选择器是直接将HTML标签作为选择器,可以是p.

千兆网络中——使用五类线、超五类线和六类线的区别

去年入手了一台Seagate的Central.一直没时间好好研究,直接插上就用了.内网速度一直维持在10M-15M左右.今年618换了款千兆路由器,速度瞬间提升到了50M-70M.线缆用的都是五类,心中就产生疑问:如果换成超五类线或者六类线,性能是不是能再提高点??于是网上搜到一个哥们十分专业的回答,忍不住粘下来跟大家共享: 原帖地址:http://we.poppur.com/thread-1098485-1-1.html 5类线的传输频率标准是100MHz  超5一般是125MHz  6类标准