自定义 Java ClassLoader

首先看一下 java.lang.ClassLoader 方法 loadClass(String name) 方法的源码

public Class<?> loadClass(String name) throws ClassNotFoundException {
        return loadClass(name, false);
}

protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            Class c = findLoadedClass(name);//检查,class是否已经被加载过了,如果加载过了就不会再加载,防止重复加载。
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    if (parent != null) {
                        c = parent.loadClass(name, false);//如果没有被加载,则交给父类 ClassLoader 进行加载,java的双亲委派类加载机制
                    } else {
                        c = findBootstrapClassOrNull(name);//如果没有父类加载器,则 直接交给最顶层的 类加载器进行加载。
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }
                if (c == null) {//如果父类加载器,也没有加载到,调用 findClass 方法。
                    // If still not found, then invoke findClass in order
                    // to find the class.
                    long t1 = System.nanoTime();
                    c = findClass(name);//findClass 源码,见下面。
                    // this is the defining class loader; record the stats
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
    /**
     * 关于方法 resolveClass 的解释
     * Links the specified class.  This (misleadingly named) method may be
     * used by a class loader to link a class.  If the class <tt>c</tt> has
     * already been linked, then this method simply returns. Otherwise, the
     * class is linked as described in the "Execution" chapter of
     * <cite>The Java&trade; Language Specification</cite>.
     * </p>
     *
     * @param  c
     *         The class to link
     *
     * @throws  NullPointerException
     *          If <tt>c</tt> is <tt>null</tt>.
     *
     * @see  #defineClass(String, byte[], int, int)
     */
                resolveClass(c);
            }
            return c;
        }
}
//什么都没有,就是简单抛出一个 Exception
protected Class<?> findClass(String name) throws ClassNotFoundException {
        throw new ClassNotFoundException(name);
}

通过分析源码,我们知道要自定义 ClassLoader 那么继承 ClassLoader 重写 findClass 方法就行了。

示例代码
import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.nio.ByteBuffer;
import java.nio.channels.Channels;
import java.nio.channels.FileChannel;
import java.nio.channels.WritableByteChannel;
public class MyClassLoader extends ClassLoader {

      @Override
      protected Class<?> findClass(String name) throws ClassNotFoundException {
            Class<?> c = null;
            try {
                  byte[] data = getClassFileBytes(name);
                  c = defineClass(name, data, 0, data.length);//name参数是要加载类的 package
            } catch (Exception e) {
                  e.printStackTrace();
            }
        return c;
      }

      private byte[] getClassFileBytes(String name) throws Exception {
        String path = "D:\\MyEclipse 10\\jd-finance-platform\\AtestJDK1-8\\bin\\";
        String classPath = name.replace(".", "\\") + ".class";
        //class 文件的存放路径
        String classFile = path + classPath;
        FileInputStream fis = new FileInputStream(classFile);
        FileChannel fileC = fis.getChannel();
        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        WritableByteChannel outC = Channels.newChannel(baos);
        ByteBuffer buffer = ByteBuffer.allocateDirect(1024);
        while (true) {
            int i = fileC.read(buffer);
            if (i == 0 || i == -1) {
                break;
            }
            buffer.flip();
            outC.write(buffer);
            buffer.clear();
        }
        fis.close();
        return baos.toByteArray();
    }

}

测试代码
public class Test{ 

    public static void main(String[] args) throws Exception{ 
      MyClassLoader myc = new MyClassLoader();
      Class<?> c = myc.loadClass("com.lp.jdk18.JDK18");//字符串是要加载类的 package
      Object obj = c.newInstance();
      System.out.println(obj);
      System.out.println(obj.getClass().getClassLoader());
      System.out.println(obj.getClass().getClassLoader().getParent());
    }

}

输出结果
com.lp.jdk18.JDK18@520b368f
com.lp.classload.MyClassLoader@77f2fbff
sun.misc.Launcher$AppClassLoader@15b94ed3
Java中的ClassLoader是一个关键组件,它负责将Java类加载到JVM中。JavaClassLoader可以分为三个层次: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方法。 需要注意的是,JavaClassLoader是一个层级结构,类的加载过程会从上至下依次进行,因此我们需要根据具体的需求来选择ClassLoader的层次。在自定义ClassLoader时,我们需要保证其所在的类路径与被加载的类所在的类路径相对应,否则就会出现ClassNotFoundException。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值