注册 登录  
 加关注
   显示下一条  |  关闭
温馨提示!由于新浪微博认证机制调整,您的新浪微博帐号绑定已过期,请重新绑定!立即重新绑定新浪微博》  |  关闭

一线天色 天宇星辰

天下武功,唯快不破

 
 
 

日志

 
 

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

2011-11-14 14:02:26|  分类: JVM |  标签: |举报 |字号 订阅

  下载LOFTER 我的照片书  |

我们知道,当一个类被加载、连接和初始化之后,他的生命周期就开始了,当该类的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以便于销毁对象,执行结果是是什么样子的呢?

< xmlnamespace prefix ="v" ns ="urn:schemas-microsoft-com:vml" />

深入理解ClassLoader(五)—类的卸载 - 一线天色 天宇星辰 - 一线天色 天宇星辰

 我们看到,前三次打印对象的

hashCode是一致的,说明是一个对象,但是最后一次却反生了变化,说明对象的地址变化了,也就是之前被卸载了(因为我们知道,类的class对象在整个类加载过程中只会被加载一次,只有一个实例对象,除非类被卸载了),然后又被加载了一次(加载了2次)。

我们再来看这几个类的关系,我们把它画成图,如下

深入理解ClassLoader(五)—类的卸载 - 一线天色 天宇星辰 - 一线天色 天宇星辰

 栈空间变量指向堆控件的数据,已经没什么争论,之前(上几篇文章)已经讨论过,

loader1对象可以通过findClass方法拿到对应的TestCase1类,因此有一个指针指向TestCase1Class实例,而TestCase1可以通过getClassLoader的方法拿到对应的类加载器,因此有一个指针指向MyClassloader的对象,TestCase1的实例obj通过getClass方法可以拿到对应的TestCase1实例。 我们这样可以看到,MyClassloader对象有2个引用指针,TestCase13个引用指针,当cloder1nullobjClassnullobjnull的时候,系统还剩下MyClassloaderTestCase1的一个相互引用。

深入理解ClassLoader(五)—类的卸载 - 一线天色 天宇星辰 - 一线天色 天宇星辰

 我们知道,只要

class引用不在有,那么这个类就可以回收了,指向方法区的就不看了,但是MyClassloaderTestCase1每个还有一个引用么人这两个其实已经没有其他引用了,但是从程序结果我们可以看到,类已经被卸载了,说明JVM已经对这种双向引用也做了卸载和回收操作,但是按照我们之前的理论,只有当引用计数器为0的时候才会被回收,这又如何理解呢?这个我们会在下面的文章,深入理解JVM—垃圾回收机制做详细介绍,敬请期待哦!

 本文对你有用么?  

票数:13 投票时间:2011-11-14 14:02:26 到 2014-12-31 23:00:00

  •     12(92.3%)
  •     1(7.7%)
  •     0(0%)
查看参与情况
编辑投票|删除投票

最新投票|博友投票

      评论这张
     
    阅读(5026)| 评论(2)
    推荐 转载

    历史上的今天

    评论

    <#--最新日志,群博日志--> <#--推荐日志--> <#--引用记录--> <#--博主推荐--> <#--随机阅读--> <#--首页推荐--> <#--历史上的今天--> <#--被推荐日志--> <#--上一篇,下一篇--> <#-- 热度 --> <#-- 网易新闻广告 --> <#--右边模块结构--> <#--评论模块结构--> <#--引用模块结构--> <#--博主发起的投票-->
     
     
     
     
     
     
     
     
     
     
     
     
     
     

    页脚

    网易公司版权所有 ©1997-2017