广告
返回顶部
首页 > 资讯 > 后端开发 > Python >自定义类加载器以及打破双亲委派模型解析
  • 154
分享到

自定义类加载器以及打破双亲委派模型解析

自定义类加载器双亲委派模型 2022-11-21 22:11:13 154人浏览 八月长安

Python 官方文档:入门教程 => 点击学习

摘要

目录1 自定义类加载器2 打破双亲委派模型1 自定义类加载器自定义类加载器的代码很简单,只需要继承ClassLoader类,覆写findClass方法即可其默认实现是会抛出

1 自定义类加载器

自定义类加载器的代码很简单,只需要继承ClassLoader类,覆写findClass方法即可

其默认实现是会抛出一个异常:

import java.io.FileInputStream; public class MyClassLoader extends ClassLoader {     private String classPath;     public MyClassLoader(String classPath) {        this.classPath = classPath;    }     private byte[] loadByte(String name) throws Exception {        name = name.replaceAll("\\.", "/");        FileInputStream fis = new FileInputStream(classPath + "/" + name + ".class");        int len = fis.available();        byte[] data = new byte[len];        fis.read(data);        fis.close();        return data;    }     @Override    protected Class<?> findClass(String name) {        byte[] data = new byte[0];        try {            data = loadByte(name);        } catch (Exception e) {            e.printStackTrace();        }        return defineClass(name, data, 0, data.length);    }}

这里是会读取指定的类路径classPath下的class文件。

相应的测试代码如下所示:

public class MyClassLoaderTest {     public static void main(String[] args) throws Exception {        MyClassLoader classLoader = new MyClassLoader("D:/test");        Class clazz = classLoader.loadClass("com.hys.test.User");        System.out.println(clazz.getClassLoader().getClass().getName());    }}

这里以User类代码为例,将其class文件放到D:/test指定目录下:

随后需要注意的是,需要将当前工作空间中的User.java文件删除。

如果不删除,根据双亲委派模型,该类会由AppClassLoader来加载,不会由自定义的的MyClassLoader来进行加载,最后运行测试代码

结果如下:

2 打破双亲委派模型

在像一些Tomcat源码中,WEBappClassLoader会打破双亲委派机制。这里我们也来简单模拟一下。

实现代码依然很简单,只需要在上述MyClassLoader类中覆写loadClass方法即可,如下:

@Overrideprotected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {    synchronized (getClassLoadingLock(name)) {        Class<?> c = findLoadedClass(name);        if (c == null) {            c = findClass(name);        }        if (resolve) {            resolveClass(c);        }        return c;    }}

这里loadClass方法的代码使用的是父类ClassLoader的源码,然后把其中使用双亲委派的代码删掉,这样MyClassLoader不用再向上去找类加载器,只会在本类中处理,这样就打破了双亲委派模型。

然后因为运行时需要加载Object类,所以将Object.class文件复制到D:/test目录下

如下所示:

随后再次运行测试类

结果如下:

可以看到,java.lang包的代码禁止被自定义的类加载器加载,防止核心api被篡改。

这是Java内部的安全检查机制。这里我们这种写法是将所有的类都交由MyClassLoader来处理,所以无法加载Java核心的类库,但是Tomcat中的类加载机制只是自定义的WebappClassLoader和CommonClassLoader打破了双亲委派模型,而其上面的BootstrapClassLoader、ExtensionClassLoader和AppClassLoader仍然还是会走双亲委派的,所以不会有问题。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。

--结束END--

本文标题: 自定义类加载器以及打破双亲委派模型解析

本文链接: https://www.lsjlt.com/news/173321.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作