JVM - 实现自定义的ClassLoader就是这么简单

Java面试Offer直通车

在这里插入图片描述

Pre

JVM-白话聊一聊JVM类加载和双亲委派机制源码解析


自定义类加载器

我们了解了双亲委派机制后,那自定义类加载器相对就很容易了 , 只需要继承 java.lang.ClassLoader 类 ,重写findClass方法即可

该类有两个核心方法:

  • 一个是loadClass(String, boolean),实现了双亲委派机制 .

  • 一个方法是findClass,默认实现是空方法

所以自定义类加载器主要是重写findClass方法


演示

在这里插入图片描述


Step1 : 复制一个Boss1 的类,编译后,取到Boss1.class 备用


Step2 : 删除Boss1类


Step3 : 编写自定义ClassLoader

在这里插入图片描述

抽象类CLassLoader ,我们自定义的ClassLoader只需要继承抽象类ClassLoader,重写loadClass方法

package com.gof.facadePattern;

import java.io.FileInputStream;
import java.lang.reflect.Method;

/**
 * @author 小工匠
 * @version v1.0
 * @create 2020-06-11 23:09
 * @motto show me the code ,change the word
 * @blog https://artisan.blog.csdn.net/
 * @description
 **/

public class MyClassLoaderTest {

    static 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) throws ClassNotFoundException {
            try {
                byte[] data = loadByte(name);
                //defineClass将一个字节数组转为Class对象,这个字节数组是class文件读取后最终的字节数组。
                return defineClass(name, data, 0, data.length);
            } catch (Exception e) {
                e.printStackTrace();
                throw new ClassNotFoundException();
            }
        }
    }

    public static void main(String args[]) throws Exception {
        //初始化自定义类加载器,会先初始化父类ClassLoader,其中会把自定义类加载器的父加载器设置为应用程序类加载器AppClassLoader
        MyClassLoader classLoader = new MyClassLoader("D:/artisan");
        //D盘创建 artisan/com/gof/facadePattern 目录,将Boss类的复制类Boss1.class丢入该目录
        Class clazz = classLoader.loadClass("com.gof.facadePattern.Boss1");
        Object obj = clazz.newInstance();
        // 调用sout方法
        Method method = clazz.getDeclaredMethod("sout", null);
        method.invoke(obj, null);
        System.out.println(clazz.getClassLoader().getClass().getName());
    }
}

defineClass 复用ClassLoader的即可 ,主要功能是将一个字节数组转为Class对象

自定义类加载器的父加载器是AppClassLoader , 但并不是说自定义ClassLoader的父类是AppClassLoader,这一点一定不要搞错了。


Step 4: 自定义目录存放Boss1.class

在这里插入图片描述


Step 5 : 运行结果

在这里插入图片描述


注意事项

Boss1 生成class后,需要把Boss1 删掉,不然双亲委派(我们并没有重写loadClass方法),它又从AppClassLoader加载了 。 需要确保你自定义加载的Boss1 在其父加载器中都不存在。

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Java中的ClassLoader是一个关键组件,它负责将Java类加载到JVM中。Java的ClassLoader可以分为三个层次:Bootstrap ClassLoader、Extension ClassLoader和Application ClassLoader自定义ClassLoader可以使得我们更好地控制Java类的加载过程,例如可以从特定的路径或者网络中加载类。 下面是一个简单自定义ClassLoader的示例代码: ```java public class MyClassLoader extends ClassLoader { private String classPath; public MyClassLoader(String classPath) { this.classPath = classPath; } @Override protected Class<?> findClass(String name) throws ClassNotFoundException { byte[] data = loadClassData(name); return defineClass(name, data, 0, data.length); } private byte[] loadClassData(String name) { String fileName = classPath + File.separatorChar + name.replace('.', File.separatorChar) + ".class"; try { FileInputStream fis = new FileInputStream(fileName); ByteArrayOutputStream baos = new ByteArrayOutputStream(); int len; byte[] buffer = new byte[1024]; while ((len = fis.read(buffer)) != -1) { baos.write(buffer, 0, len); } return baos.toByteArray(); } catch (Exception e) { e.printStackTrace(); } return null; } } ``` 上述代码中,我们继承了ClassLoader类,并实现了findClass方法,在该方法中,我们可以根据自己的需求去加载Java类。在示例中,我们从指定的路径中加载类的字节码文件,并将其转换为字节数组,最后调用defineClass方法生成Class对象。注意,这里的路径需要与ClassLoader所在的类路径相对应。 我们可以通过以下代码来使用自定义ClassLoader: ```java MyClassLoader myClassLoader = new MyClassLoader("/path/to/class/files"); Class<?> clazz = myClassLoader.loadClass("com.example.Test"); Object obj = clazz.newInstance(); Method method = clazz.getMethod("hello"); method.invoke(obj); ``` 上述代码中,我们通过自定义ClassLoader加载了Test类,并调用了hello方法。 需要注意的是,Java的ClassLoader是一个层级结构,类的加载过程会从上至下依次进行,因此我们需要根据具体的需求来选择ClassLoader的层次。在自定义ClassLoader时,我们需要保证其所在的类路径与被加载的类所在的类路径相对应,否则就会出现ClassNotFoundException。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小小工匠

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值