听到ClassLoader一脸萌?那你应该看看这篇了,比你想的简单!

图片发自简书App

想要了解更多的我?请进入我的主页查看关于我的更多详细资料。

参考资料

参考资料;
包括热修复;

背景介绍

Java程序是由一个个.class文件组成的,在程序执行阶段,JVM采用了动态加载的策略,当一个类被使用时,将由一个ClassLoad实例去加载这个.class文件。Java允许我们从外部加载一个类到内存中,然后使用它。

有关ClassLoader

Java中ClassLoader的分类

(1) BootstarpClassLoad:启动类加载器,它负责在程序启动时去加载Java的核心库。
(2) ExtensioClassLoad:扩展类加载器,它负责加载扩展库。
(3) AppClassLoad:系统类加载器,它负责加载classpath下的.class文件。
(4) CustomClassLoader:自定义的ClassLoader

ClassLoader的加载机制

ClassLoader采用了双亲委托加载机制 ,先看看下面的图解。

ClassLoader加载机制

  1. 可以看到,ClassLoader在加载一个类时,会先自下而上的检查目标时候被加载了;
  2. 然后,自上而下的依次尝试去加载目标,如果到最后一层ClassLoader仍然没有加载到,就会抛出ClassNotFoundException 错误。
  3. 注意图中每种类型ClassLoader负责的范围。

开始自定义ClassLoader

认识关键方法

  • findClass(String name):这个方法顾名思义负责查找一个类,并返回它。对我们自定义而言,这是我们最需要关注的,一般情况下,我们只需要直接在这个方法中返回目标类就可以了,这也是Google推荐我们的做法。
  • loadClass(String name):这个方法中主要负责协调加载类,通常它的逻辑比较固定,我们可以不去重写。在这个方法中,先尝试通过父类ClassLoader去加载目标类,没有加载到,然后调用findClass()方法去查找。
  • defineClass(String name, byte[] b, int off, int len):负责定义类,这个方法我们主要调用就好了。

看看例子

public class CustomClassLoader extends ClassLoader {

  private String classPath;

  public CustomClassLoader(String classPath) {
    super(CustomClassLoader.class.getClassLoader());
    this.classPath = classPath;
  }

  @Override
  public Class<?> findClass(String name) throws ClassNotFoundException {
     //检查路径是否可用
    if (classPath == null || classPath.equals("")) {
      throw new IllegalArgumentException("Please set class path first.");
    }
    //加载class文件数据
    byte[] classData = loadClassData(classPath);
    if (classData == null) {
      throw new NullPointerException(
          "Try to get the byte[] that read from class file, but mate some problem. Please check class file path.");
    }
    return defineClass(name, classData, 0, classData.length); // 将class的字节数组解码为Class实例
  }

  /**
   * 读取Class文件。这就是一个读文件的操作嘛!
   */
  private byte[] loadClassData(String path) {
    byte[] bytes = new byte[1024];
    int length = 0;
    ByteArrayOutputStream baos = new ByteArrayOutputStream();
    File classFile = new File(path);
    FileInputStream fis = null;
    try {
      fis = new FileInputStream(classFile);
      while ((length = fis.read(bytes)) != -1) {
        baos.write(bytes, 0, length);
        baos.flush();
      }
      return baos.toByteArray();
    } catch (Exception e) {
      e.printStackTrace();
    } finally {
      try {
        if (fis == null) {
          throw new NullPointerException(
              "Can not create FileInputStream, please check the file path.");
        }
        fis.close();
      } catch (IOException e) {
        e.printStackTrace();
      }
    }
    return null;
  }

  public void setClassPath(String classPath) {
    this.classPath = classPath;
  }

}

//下面来看看怎么使用
public class MyClass {
  public static void main(String[] args){
    CustomClassLoader classLoader = new CustomClassLoader("");
    try {
      classLoader.setClassPath("/Users/.../TestClass.class");
      Class clazz = classLoader.loadClass("TestClass");
      //通过反射来调用他的方法
      Method method = ReflectUtils.getMethod(clazz,"doSomething");
      System.out.println(clazz.getSimpleName());
      System.out.println("result = " + method.invoke(clazz.newInstance()));
    } catch (Exception e) {
      e.printStackTrace();
    }
  }
}

//下面是TestClass
public class TestClass {
  public int doSomething(){
    return 998;
  }
}

//下面是输出结果
TestClass
result = 998

JVM如何判断两个类是否相同

  1. 是否具有相同类名;
  2. 是否由同一个ClassLoader加载。

总结

其实Java已经做了很多工作,我们自定义ClassLoader要做的事就一件:读取.class文件 。其余的通常不需要修改。

想要了解更多的我?请进入我的主页查看关于我的更多详细资料。

作者利用空余时间分享知识不容易,看完给个赞,加个关注吧。如果你要请我吃辣条,请用打赏砸我吧🤑

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

推荐阅读更多精彩内容