什么是热加载--自定义加载器

一、什么是热加载

简单来说,就是不重启一个项目,使得部分代码更新,原理是通过java类加载器实现的。由于类的加载缺少监控使得安全性不能得到保证,一般使用在开发环境,加快开发效率。俗称开发者模式。实现自定义加载器前,先讲一讲JVM的类加载器。

二、JVM的加载器

1、BootStrap ClassLoader

BootStrap ClassLoader又叫启动类加载器,是最顶级的父类加载器,加载最核心的类,,如java.lang.、java.uti.等; 这些类位于$JAVA_HOME/jre/lib/rt.jar;

2、Extension ClassLoader

Extension ClassLoader 又叫扩展类加载器,负责加载%JAVA_HOME%/jre/lib/ext/目录下的扩展包

3、Application ClassLoader

Application ClassLoader又叫应用类加载器,负责加载classpath也就是环境变量下的类

类加载机制

工作原理:类加载采用双亲委派机制,简单来说,一个应用被加载时首先判断是否被加载过,若无则进行加载,并判断是否有父类,若有则要求父类进行加载,到顶级父类时再根据包的位置进行加载,不满足则要求子类加载。
好处:可以避免重复加载,也可以保证类加载的安全性,若自己写了个String包,那是否会替换jdk包中的String类呢?显然不可能,因为使用了这种机制来避免这种问题。

protected Class<?> loadClass(String name, boolean resolve)
    throws ClassNotFoundException
{
    synchronized (getClassLoadingLock(name)) {
        // 首先判断该类是否已经被加载过
        Class<?> c = findLoadedClass(name);
        if (c == null) {
            long t0 = System.nanoTime();
            try {
                //尝试让父类加载
                if (parent != null) {
                    c = parent.loadClass(name, false);
                } else {
                    c = findBootstrapClassOrNull(name);
                }
            } catch (ClassNotFoundException e) {
                // ClassNotFoundException thrown if class not found
                // from the non-null parent class loader
            }

            if (c == null) {
                //自定义加载,findClass是个空方法就是让我们重写的
                long t1 = System.nanoTime();
                c = findClass(name);

                // this is the defining class loader; record the stats
                sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                sun.misc.PerfCounter.getFindClasses().increment();
            }
        }
        if (resolve) {
            resolveClass(c);
        }
        return c;
    }
}

JVM是如何判断一个类是否被加载过的呢?通过findLoadedClass(name),即 java虚拟机已将此加载器记录为具有给定二进制名称的某个类的启动加载器,则返回该二进制名称的类。否则,返回null。

三、自定义加载器

首先新建一个类并继承ClassLoader。并重写findClass()方法。注释已写方法用处。

public class MyClassLoader extends ClassLoader {
@Override
//通过名称找到具体class文件,并进行加载
protected Class<?> findClass(String name) {
    byte[] b = loadClassData(name);
    name = name.split("\\.")[1];
    //然后通过defineClass()将二进制文件转化为Class对象
    return super.defineClass(name, b, 0, b.length);
}
 //这个方法就是将具体位置的class文件转化为二进制流
private byte[] loadClassData(String name) {
    try {
        name = name.replace(".", "\\");
        FileInputStream fileInputStream = new FileInputStream(new File("D:\\IntelliJ IDEA 2019.1\\JavaHotFix\\" + name+".class"));
        ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
        int i = 0;
        while ((i = fileInputStream.read()) != -1) {
            byteArrayOutputStream.write(i);
        }
        fileInputStream.close();
        return byteArrayOutputStream.toByteArray();
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }
    return null;
}

}
随便定义一个对象并定义一个方法

public class People {
    public void hello() {
        System.out.println("我是People");
    }
}

进行测试

public class Main {
public static void main(String[] args) {
    while (true) {
        try {
            MyClassLoader myClassLoader = new MyClassLoader();
            Class<?> clazz = myClassLoader.loadClass("src.People");
            Object people =  clazz.newInstance();
            clazz.getMethod("hello").invoke(people);
            Thread.sleep(2000);
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }  catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }

    }
}
}

启动后:
由于就在本级目录,先进行



启动后,改变People的hello()方法再重新一遍javac
便得到



项目结构:

小bug:由于一开始直接name直接写People导致反复不能热加载,后打断点才发现,这个加载器根本没使用到,原来限定名为People导致直接在classpath就被加载了,也就是被父类AppClassLoader加载了。

谢谢大家,觉得不错的兄弟可以把不错打在公屏上。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容