JVM之自定义类加载器入门

在上一篇博客中写到双亲委派机制.传送门
了解完双亲委派机制那么对JVM类加载器原理有一定的了解,那么书写定义类加载加载器就不会太难了.主要是重写ClassLoader中findClass方法
这个方法主要是根据包名查找class文件

package com.bonc.jvm;

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

/**
 * @Classname MyClassLoadTEst
 * @Description TODO
 * @Date 2020/6/14 16:31
 * @Created by sz
 */
public class MyClassLoadTest {

    static  class  MyClassLoad  extends  ClassLoader{

        private  String classPath;

        public MyClassLoad(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;
        }
        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 {
        MyClassLoad classLoader = new MyClassLoad("D:/tmp");
        Class clazz = classLoader.loadClass("com.bonc.jvm.User1");
        Object obj = clazz.newInstance();
        Method method = clazz.getDeclaredMethod("print", null);
        method.invoke(obj,null);
        System.out.println(clazz.getClassLoader().getClass().getName());
    }

}

package com.bonc.jvm;

import java.io.PrintStream;

public class User1
{
  private int age;
  
  public void print()
  {
    System.out.println("自定义ClassLoad");
  }
}

运行结果:
自定义类加载器运行结果
执行前提:当前classpath没有那个class文件
那么加在自定义加载器之后,整个类加载机制就更新为
有自定义加载器的双亲委派机制流程图
验证几个加载器之间的关系
注意:MyClassLoad AppClassLoader ExtClassLoader 之间都不存在继承关系.只是MyClassLoad 的parent属性是AppClassLoader 并且AppClassLoader 的parent属性是ExtClassLoader

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我可以为您提供一个简单的自定义类加载器的实现,可以加载指定位置的指定类到JVM中。请注意,由于安全性考虑,Java中的类加载器通常不允许加载系统类库或者核心类库,因此下面的示例只能加载自定义的类。 首先,我们需要创建一个继承自ClassLoader的自定义类加载器,重写findClass方法: ```java import java.io.*; public class CustomClassLoader extends ClassLoader { private String classPath; public CustomClassLoader(String classPath) { this.classPath = classPath; } @Override protected Class<?> findClass(String name) throws ClassNotFoundException { try { byte[] classBytes = getClassBytes(name); if (classBytes == null) { throw new ClassNotFoundException(name); } else { return defineClass(name, classBytes, 0, classBytes.length); } } catch (IOException e) { throw new ClassNotFoundException(name); } } private byte[] getClassBytes(String name) throws IOException { String classFilePath = classPath + File.separatorChar + name.replace('.', File.separatorChar) + ".class"; InputStream in = new FileInputStream(classFilePath); ByteArrayOutputStream out = new ByteArrayOutputStream(); byte[] buffer = new byte[1024]; int len = -1; while ((len = in.read(buffer)) != -1) { out.write(buffer, 0, len); } out.close(); in.close(); return out.toByteArray(); } } ``` 上面的代码中,我们定义了一个成员变量classPath,表示类文件所在的路径;重写了findClass方法,根据类名查找类文件并读取其内容,返回一个字节数组,然后使用defineClass方法将字节数组转换成Class对象。 然后,我们可以使用上面的自定义类加载器加载指定位置的指定类: ```java public class Test { public static void main(String[] args) throws Exception { String classPath = "/path/to/class/files"; String className = "com.example.MyClass"; CustomClassLoader loader = new CustomClassLoader(classPath); Class<?> clazz = loader.loadClass(className); Object obj = clazz.newInstance(); // do something with obj } } ``` 在上面的代码中,我们首先指定了类文件所在的路径和类名,然后创建一个自定义类加载器,并使用它加载指定的类。最后,我们可以使用newInstance方法创建一个对象,并进行一些操作。 需要注意的是,如果我们要在Java虚拟机中加载自定义的类,必须使用自定义类加载器。否则,JVM将无法识别我们的类。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值