Java中的ClasLoader之自定义ClassLoader

自定义ClassLoader

在大多数情况下,如果文件已经存在于文件系统中,内置的ClassLoader就足够用了。

但是,在需要从本地硬盘驱动器或者网络中的加载类的情况下,我们就可能需要自定义一个ClassLoader。

自定义ClassLoader不仅可以在运行时加载类,还有一些其他用例:

  1. 修改现有的字节码
  2. 动态创造符合用户需求的类。例如:在JDBC中,通过动态的加载类完成不同驱动之前的切换
  3. 在为具有相同名称和程序包的类加载不同的字节码时,实现类版本控制机制。可以通过URLClassLoader或者自定义ClassLoader来实现。

 

创建一个自定义的ClassLoader

假设,我们现在需要使用自定义的ClassLoader从文件中加载一个类,我们需要扩展ClassLoader类并重新findClass方法:

自定义的ClassLoader

在上面的例子中,我们定义了一个ClassLoader,该ClassLoader扩展了默认的ClassLoader,并从指定文件中加载字节数组。

新建一个Test类,打印“你好,帅小伙!”:

 

我们使用自定义的ClassLoader去加载它:

 

或许你可能觉得有点脱裤子放屁,直接new它香吗?这里我们仅仅是演示一下怎么去自定义一个ClassLoader,并可以正常工作。

了解java.lang.ClassLoader

loadClass(String name, boolean resolve)方法

此方法负责加载指定名称参数的类,name参数完全限定的类名称。

JVM调用loadClass方法来创建类的引用,当resolve设置为true时。但是,不一定总是要解析一个类,如果只是确定该类是否存在,则将resolve参数设置为false。

该方法是ClassLoader的入口。

我们从java.lang.ClassLoader的源码中了解loadClass()的工作原理:

 

该方法的默认实现是按以下顺序搜索类:

  1. 调用findLoadedClass(String)方法来查看是否已经加载了该类
  2. 调用父类ClassLoader的loadClass方法
  3. 调用findClass方法查找类

defineClass(String name, byte[] b, int off, int len)方法

此方法负责将字节数组转换为类的实例,在使用该类之前,我们需要调用它。

如果数据补包含有效的类,则会抛出ClassFormatError。

另外,此方法表标记为final,因此我们无法覆盖此方法。

findClass(String name)方法

此方法查找以标准名称作为参数的类,我们需要在遵循委托模型的自定义的ClassLoader中重写此方法。

另外,如果父类ClassLoader找不到请求的类,则loadClass()会调用此方法。

如果没有任何ClassLoader的父类找到该类,则引发ClassNotFoundException。

getParent()方法

此方法是尝试查找给定名称的资源。

它将首先委托给资源的父类ClassLoader,如果父级为null,则搜索JVM内置的ClassLoader的路径。

如果失败,则该方法将调用FindResource来查找资源,参数可以为相对于类路径的名称,也可以是绝对路径的名称。

它返回用于读取资源的URL对象,如果找不到资源或调用者没有足够的权限来返回资源,则返回null。

Java默认从classpath中加载资源,Java中的资源加载与位置无关,因为只要设置了环境来查找资源,代码在任何位置都无关紧要。

Context Classloaders

通常,Context ClassLoasers是为J2SE中引入的类加载委托方案提供的一种替代方法。

JVM中ClassLoader遵循分层模型,因此每个ClassLoader都有一个父类,BootstapClassLoader除外。但是,有时当JVM核心类需要动态地去加载开发人员提供的类时,我们就可能会有到问题。

例如,在JNDI中,核心功能由rt.jar中的引导程序类实现,但是这些JNDI类可能会加载由独立供应商实现的JNDI提供程序。这种情况要BootstrapClassLoader加载对AppClassLoader可见的类。

J2SE委托在这里不起作用,为了解决这个问题,我们需要找到替代的类加载方式。并且可以使用Thread Context ClassLoader来实现。

java.lang.Thread类由一个getContextClassLoader()方法,返回该线程的ContentClassLoader。该ContextClassLoader由线程的创造者提供。如果未设置,则默认为父线程的ContextClassLoade。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JavaClassLoader是一个关键组件,它负责将Java类加载到JVMJavaClassLoader可以分为三个层次: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、付费专栏及课程。

余额充值