Java 自定义类加载器实现插件式开发

最近接触Solr比较多,感觉Solr提供的插件式开发方式很酷,Solr对开发者提供了一个核心api jar包,开发者如果想扩展Solr某一项功能 比如 中文分词,只需要继承Solr提供的分词接口添加自己的实现,然后把自己的分词jar包拷贝到Solr指定目录,并在solr配置文件中配置,重启即可生效。


本文会涉及到自定义类加载,所以先介绍一下java类加载器的原理和工作机制,熟悉的同学可以直接跳过。


java类加载器

类加载器是一个用来加载类文件的类。Java源代码通过javac编译器编译成类文件。然后JVM来执行类文件中的字节码来执行程序。类加载器负责加载文件系统、网络或其他来源的类文件。有三种默认使用的类加载器:Bootstrap类加载器、Extension类加载器和System类加载器(或者叫作Application类加载器)。每种类加载器所负责加载的类也各不相同。
JVM三种预定义类型类加载器

  1. Bootstrap ClassLoader : 将存放于<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中的,并且是虚拟机识别的(仅按照文件名识别,如 rt.jar 名字不符合的类库即使放在lib目录中也不会被加载)类库加载到虚拟机内存中。启动类加载器无法被Java程序直接引用
  2. Extension ClassLoader : 将<JAVA_HOME>\lib\ext目录下的,或者被java.ext.dirs系统变量所指定的路径中的所有类库加载。开发者可以直接使用扩展类加载器。
  3. Application ClassLoader : 负责加载用户类路径(ClassPath)上所指定的类库,开发者可直接使用。

类加载器的代理模式

类加载器在尝试自己去查找某个类的字节代码并定义它时,会先代理给其父类加载器,由父类加载器先去尝试加载这个类,依次类推。在介绍代理模式背后的动机之前,首先需要说明一下 Java 虚拟机是如何判定两个 Java 类是相同的。Java 虚拟机不仅要看类的全名是否相同,还要看加载此类的类加载器是否一样。只有两者都相同的情况,才认为两个类是相同的。即便是同样的字节代码,被不同的类加载器加载之后所得到的类,也是不同的。比如一个 Java 类 com.example.Sample,编译之后生成了字节代码文件 Sample.class。两个不同的类加载器 ClassLoaderA和 ClassLoaderB分别读取了这个 Sample.class文件,并定义出两个 java.lang.Class类的实例来表示这个类。这两个实例是不相同的。对于 Java 虚拟机来说,它们是不同的类。试图对这两个类的对象进行相互赋值,会抛出运行时异常 ClassCastException。


一句话总结就是:查找某个类的时候从下至上,加载某个类的时候从上至下。


自定义类加载器

因为我们需要加载指定路径下的jar文件,所以我们需要自定义类加载器来扫描指定路径下的jar包,代码如下:

package com.bytebeats.switcher.core;

import com.bytebeats.switcher.util.IoUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.File;
import java.io.FileFilter;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;

/**
 *
 * @author Ricky Fung
 * @creat
  • 6
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
Spring Boot是一个用于构建Java企业级应用的开发框架,为开发者提供了自动配置、快速开发、便捷部署等功能。对于类加载器的使用,Spring Boot提供了默认的类加载器机制,但也可以通过自定义类加载器实现一些特殊的需求。 自定义类加载器可以通过继承ClassLoader类来实现。在自定义类加载器中,我们可以重写findClass方法,在该方法中实现自己的类加载逻辑。例如,可以从特定的位置加载类文件,或者从其他资源中加载类。通过自定义类加载器,我们可以灵活地加载一些不在常规位置的类文件或资源。 在Spring Boot中,可以使用自定义类加载器实现一些插件化的功能。例如,可以实现一个插件类加载器,负责加载插件模块,并将其实例化为Spring Bean。这样,在运行时我们可以动态地加载并使用一些自定义的功能模块,而不需要在编译时就将其包含在应用程序中。 另外,自定义类加载器还可以用于热部署功能的实现。通过定时或者其他方,我们可以在运行时重新加载某些类文件,以实现代码的热更新,避免了重启应用程序的操作。 需要注意的是,使用自定义类加载器需要谨慎。不正确的使用或者滥用类加载器可能会导致类冲突、内存泄漏等问题。因此,在自定义类加载器时,需要仔细考虑设计和实现,确保安全性和稳定性。 总之,Spring Boot提供了默认的类加载器机制,但也支持自定义类加载器。通过自定义类加载器,我们可以实现一些特殊的需求,例如插件化功能和热部署功能。但在使用自定义类加载器时,需要谨慎思考和设计,确保安全性和稳定性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值