java 详解类加载器的双亲委派及打破双亲委派

发布时间 - 2026-01-10 22:30:12    点击率:

java 详解类加载器的双亲委派及打破双亲委派

一般的场景中使用Java默认的类加载器即可,但有时为了达到某种目的又不得不实现自己的类加载器,例如为了达到类库的互相隔离,例如为了达到热部署重加载功能。这时就需要自己定义类加载器,每个类加载器加载各自的类库资源,以此达到资源隔离效果。在对资源的加载上可以沿用双亲委派机制,也可以打破双亲委派机制。

一、沿用双亲委派机制自定义类加载器很简单,只需继承ClassLoader类并重写findClass方法即可。如下例子:

①先定义一个待加载的类Test,它很简单,只是在构建函数中输出由哪个类加载器加载。

public class Test {

  public Test(){
    System.out.println(this.getClass().getClassLoader().toString());
  }

}

②定义一个TestClassLoader类继承ClassLoader,重写findClass方法,此方法要做的事情是读取Test.class字节流并传入父类的defineClass方法即可。然后就可以通过自定义累加载器TestClassLoader对Test.class进行加载,完成加载后会输出“TestLoader”。

public class TestClassLoader extends ClassLoader {

  private String name;

  public TestClassLoader(ClassLoader parent, String name) {
    super(parent);
    this.name = name;
  }

  @Override
  public String toString() {
    return this.name;
  }

  @Override
  public Class<?> findClass(String name) {

    InputStream is = null;
    byte[] data = null;
    ByteArrayOutputStream baos = new ByteArrayOutputStream();
    try {
      is = new FileInputStream(new File("d:/Test.class"));
      int c = 0;
      while (-1 != (c = is.read())) {
        baos.write(c);
      }
      data = baos.toByteArray();
    } catch (Exception e) {
      e.printStackTrace();
    } finally {
      try {
        is.close();
        baos.close();
      } catch (IOException e) {
        e.printStackTrace();
      }
    }
    return this.defineClass(name, data, 0, data.length);
  }

  public static void main(String[] args) {
    TestClassLoader loader = new TestClassLoader(
        TestClassLoader.class.getClassLoader(), "TestLoader");
    Class clazz;
    try {
      clazz = loader.loadClass("test.classloader.Test");
      Object object = clazz.newInstance();
    } catch (Exception e) {
      e.printStackTrace();
    } 
  }

}

二、打破双亲委派机制则不仅要继承ClassLoader类,还要重写loadClass和findClass方法,如下例子:

①定义Test类。

public class Test {
  public Test(){
    System.out.println(this.getClass().getClassLoader().toString());
  }
}

②重新定义一个继承ClassLoader的TestClassLoaderN类,这个类与前面的TestClassLoader类很相似,但它除了重写findClass方法外还重写了loadClass方法,默认的loadClass方法是实现了双亲委派机制的逻辑,即会先让父类加载器加载,当无法加载时才由自己加载。这里为了破坏双亲委派机制必须重写loadClass方法,即这里先尝试交由System类加载器加载,加载失败才会由自己加载。它并没有优先交给父类加载器,这就打破了双亲委派机制。

public class TestClassLoaderN extends ClassLoader {

  private String name;

  public TestClassLoaderN(ClassLoader parent, String name) {
    super(parent);
    this.name = name;
  }

  @Override
  public String toString() {
    return this.name;
  }

  @Override
  public Class<?> loadClass(String name) throws ClassNotFoundException {
    Class<?> clazz = null;
    ClassLoader system = getSystemClassLoader();
    try {
      clazz = system.loadClass(name);
    } catch (Exception e) {
      // ignore
    }
    if (clazz != null)
      return clazz;
    clazz = findClass(name);
    return clazz;
  }

  @Override
  public Class<?> findClass(String name) {

    InputStream is = null;
    byte[] data = null;
    ByteArrayOutputStream baos = new ByteArrayOutputStream();
    try {
      is = new FileInputStream(new File("d:/Test.class"));
      int c = 0;
      while (-1 != (c = is.read())) {
        baos.write(c);
      }
      data = baos.toByteArray();
    } catch (Exception e) {
      e.printStackTrace();
    } finally {
      try {
        is.close();
        baos.close();
      } catch (IOException e) {
        e.printStackTrace();
      }
    }
    return this.defineClass(name, data, 0, data.length);
  }

  public static void main(String[] args) {
    TestClassLoaderN loader = new TestClassLoaderN(
        TestClassLoaderN.class.getClassLoader(), "TestLoaderN");
    Class clazz;
    try {
      clazz = loader.loadClass("test.classloader.Test");
      Object object = clazz.newInstance();
    } catch (Exception e) {
      e.printStackTrace();
    }
  }

}

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


# java  # 类加载器的双亲委派及打破双亲委派  # 双亲委派  # 类加载  # JVM的类加载过程以及双亲委派模型详解  # Java虚拟机类加载器之双亲委派机制模型案例  # JVM要双亲委派的原因及如何打破它  # 自定义类加载器以及打破双亲委派模型解析  # 加载  # 重写  # 自定义  # 自己的  # 类库  # 只需  # 这就  # 可以通过  # 希望能  # 要做  # 很简单  # 写了  # 在对  # 又不  # 谢谢大家  # 但它  # 后会  # 时才  # 先让  # 很相似 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 网页设计与网站制作内容,怎样注册网站?  Laravel怎么发送邮件_Laravel Mail类SMTP配置教程  jQuery中的100个技巧汇总  打造顶配客厅影院,这份100寸电视推荐名单请查收  Win11怎么关闭透明效果_Windows11辅助功能视觉效果设置  如何快速生成高效建站系统源代码?  Laravel如何从数据库删除数据_Laravel destroy和delete方法区别  Laravel如何处理表单验证?(Requests代码示例)  米侠浏览器网页背景异常怎么办 米侠显示修复  为什么php本地部署后css不生效_静态资源加载失败修复技巧【技巧】  html5如何实现懒加载图片_ intersectionobserver api用法【教程】  Laravel数据库迁移怎么用_Laravel Migration管理数据库结构的正确姿势  成都品牌网站制作公司,成都营业执照年报网上怎么办理?  canvas 画布在主流浏览器中的尺寸限制详细介绍  如何在HTML表单中获取用户输入并结合JavaScript动态控制复利计算循环  香港服务器建站指南:免备案优势与SEO优化技巧全解析  C++时间戳转换成日期时间的步骤和示例代码  laravel怎么为应用开启和关闭维护模式_laravel应用维护模式开启与关闭方法  高端建站如何打造兼具美学与转化的品牌官网?  Win11怎么修改DNS服务器 Win11设置DNS加速网络【指南】  html5怎么画眼睛_HT5用Canvas或SVG画眼球瞳孔加JS控制动态【绘制】  详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)  ,南京靠谱的征婚网站?  Laravel如何配置和使用队列处理异步任务_Laravel队列驱动与任务分发实例  香港代理服务器配置指南:高匿IP选择、跨境加速与SEO优化技巧  千问怎样用提示词获取健康建议_千问健康类提示词注意事项【指南】  北京专业网站制作设计师招聘,北京白云观官方网站?  如何在沈阳梯子盘古建站优化SEO排名与功能模块?  HTML 中如何正确使用模板变量为元素的 name 属性赋值  Java遍历集合的三种方式  如何快速上传自定义模板至建站之星?  Win11怎么查看显卡温度 Win11任务管理器查看GPU温度【技巧】  潮流网站制作头像软件下载,适合母子的网名有哪些?  通义万相免费版怎么用_通义万相免费版使用方法详细指南【教程】  如何实现javascript表单验证_正则表达式有哪些实用技巧  QQ浏览器网页版登录入口 个人中心在线进入  详解vue.js组件化开发实践  如何批量查询域名的建站时间记录?  Laravel怎么使用Session存储数据_Laravel会话管理与自定义驱动配置【详解】  Laravel Eloquent访问器与修改器是什么_Laravel Accessors & Mutators数据处理技巧  怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?  深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?  Laravel如何使用集合(Collections)进行数据处理_Laravel Collection常用方法与技巧  JavaScript Ajax实现异步通信  今日头条AI怎样推荐抢票工具_今日头条AI抢票工具推荐算法与筛选【技巧】  如何用IIS7快速搭建并优化网站站点?  瓜子二手车官方网站在线入口 瓜子二手车网页版官网通道入口  如何在景安云服务器上绑定域名并配置虚拟主机?  Android 常见的图片加载框架详细介绍  高端云建站费用究竟需要多少预算?