重写类加载器,实现简单的热替换

一、前言

关于类加载器,前面写了三篇,这篇是第四篇。

实战分析Tomcat的类加载器结构(使用Eclipse MAT验证)

还是Tomcat,关于类加载器的趣味实验

 

本篇写个简单的例子,来说说类的热替换。

 先说个原则,在同一个类加载器内,不能重复加载同一个类(因为 classloader 在 loadClass 一次后会缓存在类加载器内部,此时如果再次加载,其实是直接从缓存取,我意思的加载,是指真正去调用 defineClass 去加载。)。所以,要热替换一个类,必须连其类加载器一起换掉。

 

 

二、步骤

1、源码

一共两个工程,工程1,只有下面这一个类

测试类,TestSample.java,这个类的用处就是,我们不断改变其 printClassLoader 的代码,并重新编译后,放到指定位置:

/**
 * desc:
 *
 * @author : caokunliang
 * creat_date: 2019/6/15 0015
 * creat_time: 14:01
 **/
public class TestSample {

    public void printClassLoader(TestSample testSample) {
        System.out.println(testSample.getClass().getClassLoader());
    }
}

 

工程2,两个类:

ReloadMainTest.java,主要是启动一个定时任务,定时任务会每隔3s,用一个自定义的类加载器,去指定位置(为了简单,直接路径写死了)加载 TestSample.class,并调用其方法进行打印,查看是否热替换成功:

import java.lang.reflect.Method;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

/**
 * desc:
 *
 * @author : caokunliang
 * creat_date: 2019/6/14 0014
 * creat_time: 17:04
 **/
public class ReloadMainTest {
    public static void reload()throws Exception{

        String className = "TestSample";
        MyClassLoader classLoader = new MyClassLoader("/home/test/TestSample.class", className);
        Class<?> loadClass = classLoader.findClass(className);
        Object instance = loadClass.newInstance();
        Method method = instance.getClass().getMethod("printClassLoader", new Class[]{loadClass});
        method.invoke(instance,instance);


    }


    public static void main(String[] args) throws Exception {
          testReload();
    }

    public static void testReload(){
        //创建一个2s执行一次的定时任务
        Executors.newScheduledThreadPool(1).scheduleAtFixedRate(new Runnable() {
            @Override
            public void run() {
                try {
                    reload();
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        },0,3, TimeUnit.SECONDS);


    }
}

 

MyClassLoader.java,自定义的类加载器:
import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.io.UnsupportedEncodingException;

/**
 * desc:
 *
 * @author : caokunliang
 * creat_date: 2019/6/13 0013
 * creat_time: 10:19
 **/
public class MyClassLoader extends ClassLoader {
    private String classPath;
    private String className;


    public MyClassLoader(String classPath, String className) {
        this.classPath = classPath;
        this.className = className;
    }

    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        byte[] data = getData();
        return defineClass(className,data,0,data.length);
    }

    private byte[] getData(){
        String path = classPath;

        try {
            FileInputStream inputStream = new FileInputStream(path);
            ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
            byte[] bytes = new byte[2048];
            int num = 0;
            while ((num = inputStream.read(bytes)) != -1){
                byteArrayOutputStream.write(bytes, 0,num);
            }

            return byteArrayOutputStream.toByteArray();
        } catch (Exception e) {
            e.printStackTrace();
        }

        return null;
    }
}

 

2、测试

我这边的测试路径为:/home/test, MyClassLoader.java已经编译

[root@localhost test]# pwd
/home/test
[root@localhost test]# ll MyClassLoader.*
-rw-r--r--. 1 root root 1175 Jun 13 11:25 MyClassLoader.class
-rw-r--r--. 1 root root 1242 Jun 13 11:25 MyClassLoader.java

 

我的工程2 的代码放在另一个目录下:

[root@localhost test-reload]# pwd
/home/test/test-reload
[root@localhost test-reload]# ll
total 20
-rw-r--r--. 1 root root 1464 Jun 15 17:55 MyClassLoader.class
-rw-r--r--. 1 root root 1458 Jun 15 17:55 MyClassLoader.java
-rw-r--r--. 1 root root  511 Jun 15 17:55 ReloadMainTest$1.class
-rw-r--r--. 1 root root 1531 Jun 15 17:55 ReloadMainTest.class
-rw-r--r--. 1 root root 1218 Jun 15 17:52 ReloadMainTest.java

 

执行 java ReloadMainTest,启动测试类,就会每个3s,执行 TestSample 的方法:

 

此时,我们在另一个窗口中,去修改 TestSample.java,并重新编译之:

 

此时,我们切回原窗口,可以发现输出发生了变化:

 

3、测试进阶

这里要介绍一个工具,阿里开源的arthas。 (https://alibaba.github.io/arthas/en/install-detail.html

这款工具,功能很强,下图是其简单介绍:

 

这里,我打算使用其 类搜索功能,通过搜索  TestSample 类,来查看该类是从哪个类加载器加载而来,使用方式极其简单,直接java 启动 arthas,然后选择要attach的java 应用。

[root@localhost test]# java -jar arthas-boot.jar 
[INFO] arthas-boot version: 3.1.1
[INFO] Found existing java process, please choose one and hit RETURN.
* [1]: 10100 org.apache.catalina.startup.Bootstrap
  [2]: 25517 ReloadMainTest
2
[INFO] arthas home: /root/.arthas/lib/3.1.1/arthas
[INFO] Try to attach process 25517
[INFO] Attach process 25517 success.
[INFO] arthas-client connect 127.0.0.1 3658
  ,---.  ,------. ,--------.,--.  ,--.  ,---.   ,---.                           
 /  O  \ |  .--. ''--.  .--'|  '--'  | /  O  \ '   .-'                          
|  .-.  ||  '--'.'   |  |   |  .--.  ||  .-.  |`.  `-.                          
|  | |  ||  |\  \    |  |   |  |  |  ||  | |  |.-'    |                         
`--' `--'`--' '--'   `--'   `--'  `--'`--' `--'`-----'                          
                                                                                

wiki      https://alibaba.github.io/arthas                                      
tutorials https://alibaba.github.io/arthas/arthas-tutorials                     
version   3.1.1                                                                 
pid       25517                                                                 
time      2019-06-15 19:16:59

 

下面我们搜索下TestSample类,(直接输入:sc -df TestSample):

 

是不是看到类加载器了,但这只是我截了一部分的图而已,这个命令会把 当前java进程中所有的匹配这个类的都搜出来。我们看看到底搜出来多少:

 

这里显示了,一共有9行,也就是说,在我们的定时器线程的不断运行下,每隔3s就用一个新的类加载器去加载 TestSample,目前java 进程中,已经有9个 TestSample 类了。

多个同名类,(但不同类加载器),会不会有问题?按理说应该不会,因为假设另一个类B引用该类,那么类B默认就会用它自己的类加载器来加载该类,按理说,是加载不到的,直接就报错了。(存疑。。。)

 

说回来(实在是编不下去了。。),这里我们的 ReloadMainTest,都是 把一个classloader 用完即弃,包括 该classloader 加载的类,以及用加载的类new出来的对象,都是在一个方法内,属于局部变量,跑完一次循环,就没人持有他们的引用了。

但是,为什么我们还看到有9个类存在呢? 这个主要还是因为,class 相关的数据都是存放在 永久代,永久代平时一般不进行垃圾回收,所以我们才能看到那些废弃类的尸体。我们可以试试调用垃圾回收,通过jmap就可以触发。

[root@localhost test]# jmap -dump:live,format=b,file=heap23.bin  25517
Dumping heap to /home/test/heap23.bin ...
Heap dump file created

 

此时再看类的数量,是不是变了:

 

 三、简单总结

这篇简单介绍了如何进行类的热替换。这里的热替换,建立在这样的基础上:我们加载了新的class,然后new了对象,调用了对象的方法后,整个过程就结束了,没涉及到和其他类的交互。正因为如此,新生成的对象没有被任何地方引用,所以可以进行垃圾回收;对象被回收后,perm区的class对象也就可以进行回收了,于是,classloader也没被任何地方引用,也可以进行回收,所以最后的那个测试才能出现上述的结果(即:jmap触发full gc后,TestSample的数量变回1)。

客观来说,暂时还没发现在真实环境里能发挥出什么作用,但是作为学习案例,是够了的。为什么在真实环境没用(比如 java web项目),在这类项目中,应用被打成一个war包(jar包的spring boot方式还没研究内部的类加载器结构,不能乱说),应用的WEB-INF下的classes和lib目录下的 jar 包,都是由同一个类加载器(也就是webappclassloader)加载。如果要替换的话,只能整个 webappclassloader 全部换掉才可能。能不能单独换一个类呢,我感觉是不行的,假设 ControllerA 里面引用了 AService,AServiceImpl实现AService,你说我现在想换掉 AServiceImpl,假设我们重新用自定义的类加载器 去某个位置加载 了新的 AServiceImpl ,那么我们要怎么才能让 AService 引用到这个新的 实现类呢? 且不说这二者由不同的类加载器加载,其次,还得把之前的旧的实现的被别处引用的地方给换掉。。。想想还是很不好搞。。。

这里预告一下,下一篇会是一个黑科技,尤其是对java web、java 后台开发人员而言,主要是给后台程序开个后门,执行我们的任意代码,在程序不重启的情况下进行调试、全局参数查看、方法执行等,给同事们演示了一下,效果还是很不错的。

 

 

posted @ 2019-06-17 23:49  三国梦回  阅读(3646)  评论(4编辑  收藏  举报