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、类卸载的例子

  1. public class MyTest161 extends ClassLoader{
  2.  
  3. private String className;
  4.  
  5. //目录
  6. private String path;
  7.  
  8. private final String fileExtension = ".class";
  9.  
  10. public MyTest161(String classLoadName){
  11. super(); //将系统类加载器当做该类加载器的父加载器
  12. this.className = classLoadName;
  13. }
  14.  
  15. public MyTest161(ClassLoader parent, String classLoadName){
  16. super(parent); //显示指定该类加载器的父加载器器
  17. this.className = classLoadName;
  18. }
  19.  
  20. public void setPath(String path) {
  21. this.path = path;
  22. }
  23.  
  24. @Override
  25. public String toString() {
  26. return "[" + this.className + "]";
  27. }
  28.  
  29. @Override
  30. protected Class<?> findClass(String clasName) throws ClassNotFoundException {
  31. System.out.println("findClass invoked:" + clasName);
  32. System.out.println("class loader name: " + this.className);
  33. byte[] data = this.loadClassData(clasName);
  34. return this.defineClass(clasName,data, 0, data.length);
  35. }
  36.  
  37. private byte[] loadClassData(String className){
  38. InputStream is = null;
  39. byte[] data = null;
  40. ByteArrayOutputStream baos = null;
  41.  
  42. try{
  43. className = className.replace(".","//");
  44. //System.out.println("className:" +this.className);
  45. is = new FileInputStream(new File(this.path + className + this.fileExtension));
  46. baos = new ByteArrayOutputStream();
  47. int ch = 0;
  48. while ( -1 != (ch = is.read())){
  49. baos.write(ch);
  50. }
  51. data = baos.toByteArray();
  52.  
  53. }catch (Exception ex){
  54. ex.printStackTrace();
  55. }finally {
  56. try {
  57. is.close();
  58. baos.close();
  59. }catch (Exception ex){
  60. ex.printStackTrace();
  61. }
  62. }
  63.  
  64. return data;
  65.  
  66. }
  67.  
  68. public static void main(String[] args) throws Exception{
  69. MyTest161 loader1 = new MyTest161("loader1");
  70. loader1.setPath("D:/temp/");
  71. Class<?> clazz = loader1.loadClass("com.example.jvm.classloader.MyTest1"); //
  72. System.out.println("class:" + clazz.hashCode());
  73. Object object = clazz.newInstance();
  74. System.out.println(object);
  75. loader1 = null;
  76. clazz = null;
  77. object = null;
  78. System.gc(); // 垃圾回收。实际开发不会用
  79. System.out.println();
  80.  
  81. loader1 = new MyTest161("loader1");
  82. loader1.setPath("D:/temp/");
  83. clazz = loader1.loadClass("com.example.jvm.classloader.MyTest1"); //
  84. System.out.println("class:" + clazz.hashCode());
  85. object = clazz.newInstance();
  86. System.out.println(object);
  87. System.out.println();
  88.  
  89. }
  90.  
  91. }

  设置JVM参数 -XX:+TraceClassUnloading

运行结果:

  1. findClass invoked:com.example.jvm.classloader.MyTest1
  2. class loader name: loader1
  3. class:21685669
  4. com.example.jvm.classloader.MyTest1@7f31245a
  5.  
  6. findClass invoked:com.example.jvm.classloader.MyTest1
  7. class loader name: loader1
  8. class:1173230247
  9. [Unloading class com.example.jvm.classloader.MyTest1 0x0000000100061028]
  10. com.example.jvm.classloader.MyTest1@330bedb4

  

7、使用JVisualVM查看类的卸载

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

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

JVM 类的卸载的更多相关文章

  1. Java虚拟机JVM学习07 类的卸载机制

    Java虚拟机JVM学习07 类的卸载机制 类的生命周期 当Sample类被加载.连接和初始化后,它的生命周期就开始了. 当代表Sample类的Class对象不再被引用,即不可触及时,Class对象就 ...

  2. JVM学习七:JVM之类加载器之类的卸载

    类加载的过程和原理,以及双亲委派机制都已经讲解完成,那么我们今天讲解类加载的最后一节,那么就是类的卸载. 我们知道,当一个类被加载.连接和初始化之后,他的生命周期就开始了,当该类的class对象不再被 ...

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

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

  4. JVM 类的生命周期、类加载器

    类的加载.连接与初始化                  • 1. 加载:查找并加载类的二进制数据         • 2. 连接             – 2.1 验证:确保被加载的类的正确性   ...

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

    类加载器的命名空间 每个类加载器又有一个命名空间,由其以及其父加载器组成 类加载器的命名空间的作用和影响 每个类加载器又有一个命名空间,由其以及其父加载器组成 在每个类加载器自己的命名空间中不能出现相 ...

  6. 【Java虚拟机8】自定义类加载器、类加载器命名空间、类的卸载

    前言 学习类加载器就一定要自己实现一个类加载器,今天就从一个简单的自定义类加载器说起. 自定义类加载器 例1 一个简单的类加载器,从一个给定的二进制名字读取一个字节码文件的内容,然后生成对应的clas ...

  7. JVM中类的卸载机制

    类的生命周期 当Sample类被载入.连接和初始化后,它的生命周期就開始了. 当代表Sample类的Class对象不再被引用,即不可触及时,Class对象就会结束生命周期.Sample类在方法区内的数 ...

  8. 深入java虚拟机学习 -- 类的卸载

    类的生命周期 在开始本节之前让我们再来回顾下类的生命周期 没看过前6个过程的同学建议从头看下<深入java虚拟机学习 -- 类的加载机制>,这里就不再过多介绍了,着重说下类的卸载 类的卸载 ...

  9. JVM类文件结构

    作为一名Java后台开发的程序员, 深入理解JVM, 重要性不言而喻, 这篇文章主要是记录JVM类文件结构相关知识. 2. 实例 这部分比较抽象, 所以以实例的形式来学习. 这部分作为资料, 以便后面 ...

随机推荐

  1. 过滤器+用session验证是否登陆过

    过滤器: public class MyActionFilter : ActionFilterAttribute//继承ActionFilterAttribute类 { public override ...

  2. java--Annotation实现

    Annotation实现 在java中一共提供了三个annotation:@Override,@Deprecated,@SupperessWarnings 代码范例:使用Annotation pack ...

  3. java操作redis(jedis)常用方法示例

    说明:redis命令和jedis方法名基本是一一对应的 Redis常用命令1 连接操作命令 ● quit:关闭连接(connection) ● auth:简单密码认证 ● help cmd: 查看cm ...

  4. 通过mock-api模拟真实数据

    一.什么是 mock-api mock-api(模客)是一个便捷的 REST 接口数据模拟工具,用于前后端隔离测试,开发调试. 二.mock-api 的使用 1.注册登录 http://mock-ap ...

  5. CentOS7怎样安装MySQL5.7.22

    wget http://dev.mysql.com/get/mysql57-community-release-el7-8.noarch.rpm下载mysql源安装包 yum install mysq ...

  6. #《你们都是魔鬼吗》第八次团队作业:第五天Alpha冲刺

    <你们都是魔鬼吗>第八次团队作业:Alpha冲刺 项目 内容 这个作业属于哪个课程 任课教师博客主页链接 这个作业的要求在哪里 作业链接地址 团队名称 你们都是魔鬼吗 作业学习目标 完成最 ...

  7. Shell排序——软考(五)

    希尔排序是一种插入排序,是对直接插入排序的一种改进,该算法出自于D.L.Shell,因此得名为希尔.Shell排序又名缩小增量排序. 思想      假设初始序列为n个元素,先取一个小于n的整数d1作 ...

  8. Redis 中的高级数据类型

    5个基础数据类型 高级功能 ① HyperLogLog (参考) 供不精确的去重计数功能,比较适合用来做大规模数据的去重统计,例如统计 UV > PFADD visitors alice bob ...

  9. 2019-2020-1 20199312《Linux内核原理与分析》第七周作业

    进程的描述 操作系统内核实现操作系统的三大管理功能 进程管理(内核中最核心的功能) 内存管理 文件系统 在操作系统中,我们通过进程控制块PCB描述进程. 为了管理进程,内核必须对每个进程进行清晰的描述 ...

  10. dede5.7评论框不显示,文章页表情不显示,解决办法

    dede5.7评论框不显示,文章页表情不显示,解决办法 进入dede后台,系统-系统基本参数-其它选项,找到模板引擎禁用标签,去掉里面的php,如图 进入dede模板目录,默认是/templets/d ...