Java动态重新加载Class

项目中使用到了动态重新加载Class的机制,作用是让一些代码上线之前可以在线上环境测试一下,当然,这是非常不好的测试机制,我刚来的时候也为这种机制感到惊讶—怎么可以在线上环境运行测试代码!后来经过了解,这么做的原因有以下两个:

  • 有些代码没有办法在本地进行测试,本地没有线上的环境
  • 我们弱到连测试机都没有(这是重点)

既然我们连测试机都没有,那么我就觉得我们的项目其实也没有想象中的重要,这么测就这么测吧~~ 
    之前对ClassLoader没啥概念,google到一篇文章,翻译了一下并且做了一些补充,加深记忆 
原文地址:

引用
http://tutorials.jenkov.com/java-reflection/dynamic-class-loading-reloading.html#classloader

--------------------------------------------------- 
ClassLoader 
    顾名思义,ClassLoader就是用来Load Class的,当一个Class被加载的时候,这个Class所引用到的所有Class也会被加载,而且这种加载是递归的,也就是说,如果A引用到B,B 引用到C,那么当A被加载的时候,B也会被加载,而B被加载的时候,C也会加载。如此递归直到所有需要的Class都加载好。 
    常见的ClassLoader:

引用
* Bootstrap class loader:虚拟机运行时必须要用到的类的加载器,比如java.*。它通常是在虚拟机种用本地代码(如C)实现,在系统中用null表示。 
* Extension class loader:负责加载ext目录下的Class。 
* Application class loader:负责加载CLASSPATH上的类。

ClassLoader的代理层次关系 
    ClassLoader是以层次关系组织起来的,当你创建一个标准的Java ClassLoader的时候,你必须提供一个父ClassLoader。当一个ClassLoader需要加载一个Class的时候,它首先会让父 ClassLoader去加载这个Class,如果父ClassLoader不能加载这个Class,那么当前的ClassLoader才会自己去加载。 
    ClassLoader加载Class的步骤:

  • 检查这个Class是否已经被加载过了
  • 如果没有被加载过,那么让父ClassLoader尝试去加载
  • 如果父ClassLoader无法加载,那么尝试使用当前ClassLoader加载

从ClassLoader加载Class的步骤可以得知,如果你需要动态重新加载一个Class,那么你的ClassLoader必须跟上述标准流程有所区别,需要动态加载的Class不能交给父ClassLoader,否则你自己的ClassLoader将没有机会去加载这个Class(因为正常情况下父ClassLoader总是能加载到你所请求的Class)。 
    所以,如果你需要ClassLoader重新加载一个Class,重写findClass方法是起不到效果的,因为findClass在父 ClassLoader加载失败之后才会执行

  1. // First, check if the class has already been loaded
  2. Class c = findLoadedClass(name);
  3. if (c == null) {
  4. try {
  5. if (parent != null) {
  6. c = parent.loadClass(name, false);
  7. } else {
  8. c = findBootstrapClass0(name);
  9. }
  10. } catch (ClassNotFoundException e) {
  11. // If still not found, then invoke findClass in order
  12. // to find the class.
  13. c = findClass(name);
  14. }
  15. }

必须重写loadClass方法才能达到效果。

动态重新加载Class 
    Java内置的ClassLoader总会在加载一个Class之前检查这个Class是否已经被加载过,已经被加载过的Class不会加载第二次。因此要想重新加载Class,我们需要实现自己的ClassLoader。 
    另外一个问题是,每个被加载的Class都需要被链接(link),这是通过执行ClassLoader.resolve()来实现的,这个方法是 final的,因此无法重写。Resove()方法不允许一个ClassLoader实例link一个Class两次,因此,当你需要重新加载一个 Class的时候,你需要重新New一个你自己的ClassLoader实例。 
    刚才说到一个Class不能被一个ClassLoader实例加载两次,但是可以被不同的ClassLoader实例加载,这会带来新的问题:

  1. MyObject object = (MyObject)
  2. myClassReloadingFactory.newInstance("com.jenkov.MyObject");

这段代码会导致一个ClassCastException,因为在一个Java应用中,Class是根据它的全名(包名+类名)和加载它的 ClassLoader来唯一标识的。在上面的代码中object对象对应的Class和newInstance返回的实例对应的Class是有区别的:

  全名 ClassLoader实例
Object对象的Class com.jenkov.MyObject  AppClassLoader实例
newInstance返回对象的Class com.jenkov.MyObject 自定义ClassLoader实例

解决的办法是使用接口或者父类,只重新加载实现类或者子类即可。

  1. MyObjectInterface object = (MyObjectInterface)
  2. myClassReloadingFactory.newInstance("com.jenkov.MyObject");
  3. MyObjectSuperclass object = ( MyObjectSuperclass)
  4. myClassReloadingFactory.newInstance("com.jenkov.MyObject");

在自己实现的ClassLoader中,当需要加载MyObjectInterface或者MyObjectSuperclass的时候,要代理给父 ClassLoader去加载。

Java_动态重新加载Class机制的更多相关文章

  1. Java_动态重新加载Class总结

    在此记载Java动态重新加载Class的点点滴滴,实现之前也在网上看了很多文章,但发现不是很清晰,后来发现总结,看源码实现还是最靠谱. 直接上代码: package com.lkb.autoCode. ...

  2. Android系统下的动态Dex加载

    1 问题在Android系统中,一个App的所有代码都在一个Dex文件里面.Dex是一个类似Jar的存储了多有Java编译字节码的归档文件.因为Android系统使用Dalvik虚拟机,所以需要把使用 ...

  3. Android系统下的动态Dex加载与app速度优化

    1 问题 在Android系统中,一个App的所有代码都在一个Dex文件里面.Dex是一个类似Jar的存储了多有Java编译字节码的归档文件.因为Android系统使用Dalvik虚拟机,所以需要把 ...

  4. 使用javassist运行时动态重新加载java类及其他替换选择

    在不少的情况下,我们需要对生产中的系统进行问题排查,但是又不能重启应用,java应用不同于数据库的存储过程,至少到目前为止,还不能原生的支持随时进行编译替换,从这种角度来说,数据库比java的动态性要 ...

  5. linux动态库加载RPATH, RUNPATH

    摘自http://gotowqj.iteye.com/blog/1926771 linux动态库加载RPATH, RUNPATH 链接动态库 如何程序在连接时使用了共享库,就必须在运行的时候能够找到共 ...

  6. esri-leaflet入门教程(5)- 动态要素加载

    esri-leaflet入门教程(5)- 动态要素加载 by 李远祥 在上一章节中已经说明了esr-leaflet是如何加载ArcGIS Server提供的各种服务,这些都是服务本身来决定的,API脚 ...

  7. 【微信小程序】模仿58同城页面制作以及动态数据加载

    完成动态数据的加载,如下 使用上班的空余时间慢慢的学习,相信总有一天我会很熟悉的掌握这门技术. 本次学习小总结: 微信小程序使用的代码基本与HTML.CSS.JS等前段有关知识一样. 微信小程序js使 ...

  8. libdl.so 动态库加载、查找

    使用libdl.so库 动态库加载原理   动态库中函数的查找已经封装成 libdl.so,有4个函数: dlopen  : 打开一个动态库 dlsym   : 在打开的动态库里找一个函数 dlclo ...

  9. Vue 动态图片加载路径问题和解决方法

    最近在做一个树形结构的组件,使用了Vue和element UI中el-tree组件.因为树中每个节点都需要显示一个图标图片,并且需要根据后台传入的数据类型动态地显示,所以图片的路径需要动态地加载.下面 ...

随机推荐

  1. [NHibernate]集合类(Collections)映射

    系列文章 [Nhibernate]体系结构 [NHibernate]ISessionFactory配置 [NHibernate]持久化类(Persistent Classes) [NHibernate ...

  2. 转 C# 只允许运行一个实例

    来源:http://blog.csdn.net/jin20000/article/details/3136791 互斥进程(程序), 简单点说,就是在系统中只能有该程序的一个实例运行. 现在很多软件都 ...

  3. [Python] 利用Django进行Web开发系列(二)

    1 编写第一个静态页面——Hello world页面 在上一篇博客<[Python] 利用Django进行Web开发系列(一)>中,我们创建了自己的目录mysite. Step1:创建视图 ...

  4. PHP常用字符串的操作函数

    字符串转换类函数 addcslashes函数:以C语言风格使用反斜线转义字符串中的字符 addslashes函数:使用反斜线引用字符串 chop函数:清除字符串中的连续空格 get_html_tran ...

  5. Daily Scrum Meeting ——TenthDay

    一.Daily Scrum Meeting照片 二.Burndown Chart 新增了几个issues 三.项目进展 1.完成了登录界面与管理员和发布者界面的整合. 2.活动发布者界面的完成 四.问 ...

  6. poj 1112

    昨天晚上看的题. 说实话,我一眼就看出了是二分图,再一眼就看出了是二分图+dp(01背包).但悲剧的是我一眼看出的算法是正确的,但我总以为它是错误的,浪费了很长时间像其他算法(TAT). 今天终于把代 ...

  7. 多重网格法简介(Multi Grid)

    原文链接 多重网格法是一种用于求解方程组的方法,可用于插值.解微分方程等. 从专业角度讲多重网格法实际上是一种多分辨率的算法,由于直接在高分辨率(用于求解的间隔小)上进行求解时对于低频部分收敛较慢,与 ...

  8. NSCache

    今天在优化的时候,用了NSCache,感觉没什么两样(视觉上).按理内存缓存,怎么也比从硬盘读取的要快.. dispatch_async(dispatch_get_global_queue(, ), ...

  9. NOIp2014 解题报告

    有史以来第一届面向社会征题的NOIp结束了.最开始以为面向社会征题会很难,但是这是我参加的最水的一次NOIp了. 由于停了两月的课,所以现在正在补文化科目就没时间打代码了.所以所有的题目就均不给出代码 ...

  10. SET基本数据类型

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAO4AAADZCAIAAACo85tgAAAgAElEQVR4Aey9SdAs13XnV/P8jW8e8D