Java基础- 自定义类加载器

自定义类加载器

在 Java 中实现自定义类加载器通常涉及继承 ClassLoader 类并重写其 findClass 方法。自定义类加载器允许我们从非标准来源(如网络、加密文件或其他媒体)加载类。下面是实现自定义类加载器的基本步骤:

1. 继承 ClassLoader

创建一个新的类并继承 ClassLoader 类。例如:

public class MyClassLoader extends ClassLoader {// 类的实现
}

2. 重写 findClass 方法

在自定义类加载器中重写 findClass 方法。这是类加载的核心,我们需要在这里实现查找类字节码并定义类的逻辑。

@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {// 实现加载类的逻辑// 比如从文件系统、网络或其他源读取类的字节码
}

3. 读取类数据

findClass 方法中,需要实现读取类的字节码数据的逻辑。这可能涉及读取文件、网络资源等。

4. 调用 defineClass 方法

一旦获得了类的字节码,使用 defineClass 方法来将这些字节码转换为 Class 对象。defineClassClassLoader 类的一个受保护方法,可以将一个字节数组转换为 Class 类的实例。

byte[] bytes = ... // 从文件或其他地方获取的类字节码
Class<?> c = defineClass(name, bytes, 0, bytes.length);

5. 返回类对象

findClass 方法的末尾,返回通过 defineClass 方法创建的类对象。

完整示例

以下是一个简单的自定义类加载器示例,它从文件系统加载类:

public class MyClassLoader extends ClassLoader {private String classPath;public MyClassLoader(String classPath) {this.classPath = classPath;}@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {try {byte[] data = loadClassData(name);return defineClass(name, data, 0, data.length);} catch (IOException e) {throw new ClassNotFoundException("Could not load class " + name, e);}}private byte[] loadClassData(String name) throws IOException {String path = classPath + name.replace('.', '/') + ".class";InputStream is = new FileInputStream(path);ByteArrayOutputStream byteStream = new ByteArrayOutputStream();int nextValue = 0;while ((nextValue = is.read()) != -1) {byteStream.write(nextValue);}return byteStream.toByteArray();}
}

在这个示例中,MyClassLoader 从指定的文件路径读取类文件,将文件内容转换为字节数组,然后使用 defineClass 方法来创建 Class 对象。

注意事项

  • 安全考虑:自定义类加载器在处理类字节码时应确保数据的来源是安全的。
  • 性能考虑:合理管理资源,如关闭打开的文件流。
  • 遵循双亲委派模型:默认情况下,类加载器会先尝试让其父类加载器加载类,以保证 Java 核心库的类不会被覆盖。如果需要打破这一模型,需谨慎操作并充分理解可能的后果。

自定义类加载器使得 Java 程序能够以灵活和动态的方式加载类,但同时需要确保代码的安全和有效性。


findClass方法

在 Java 中,findClass 方法是 ClassLoader 类的一部分,用于加载类的具体实现。这个方法在自定义类加载器中尤其重要,因为它提供了一种机制来查找和加载类,特别是当类不在标准的类加载路径中时。

基本功能

findClass 方法被设计用来在类加载器的父级加载器都未能成功加载类时,查找并加载类。当一个类被加载时,ClassLoader 类的 loadClass 方法会首先调用父类加载器尝试加载该类。如果父类加载器无法加载该类(通常因为类不在它们的搜索路径中),loadClass 方法会调用 findClass 方法来尝试加载该类。

方法签名

findClass 方法通常如下定义:

protected Class<?> findClass(String name) throws ClassNotFoundException

参数说明:

  • name:需要加载的类的完全限定名。

使用场景

  • 自定义类加载器:在创建自定义类加载器时,通常需要重写 findClass 方法。这允许开发者定义自己的类查找逻辑,比如从特定的文件路径、数据库或其他来源加载类。
  • 插件和模块化系统:在实现插件或模块化系统时,findClass 方法可以被用来从模块化的、分离的资源中加载类。

实现注意事项

  • 重写 findClass:在自定义类加载器中,通常需要重写 findClass 方法以提供新的类查找和加载机制。
  • 调用 defineClass:在 findClass 方法中,一旦类的字节码被找到和加载,通常会调用 ClassLoader 类的 defineClass 方法来将字节码转换成 Class 对象。
  • 异常处理:如果 findClass 无法找到或加载类,它应该抛出 ClassNotFoundException

示例

下面是一个简单的自定义类加载器示例,展示了如何重写 findClass 方法:

public class MyClassLoader extends ClassLoader {@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {byte[] b = loadClassData(name);return defineClass(name, b, 0, b.length);}private byte[] loadClassData(String name) {// 实现从文件系统、网络或其他来源加载类字节码的逻辑// ...}
}

在这个例子中,MyClassLoader 类重写了 findClass 方法,以提供从特定来源加载类的逻辑。一旦类的字节码被加载,它使用 defineClass 方法来创建 Class 对象。

结论

findClass 方法是自定义类加载器实现中的关键部分,它提供了一种灵活的方式来扩展 Java 的类加载机制。通过重写 findClass,开发者可以控制如何查找和加载类,这在实现插件系统、应用服务器或处理非标准类加载请求时尤其有用。


defineClass方法

在 Java 中,defineClass 方法是 ClassLoader 类的一个关键方法,用于将字节数组转换成 Class 对象。这个方法在自定义类加载器中尤其重要,因为它允许开发者从非标准来源加载类,例如从网络、加密文件等。

基本功能

defineClass 方法的主要功能是将一个字节数组(包含类的字节码)转换成一个 Class 对象。这个方法通常在自定义的类加载器中被重写或调用,以实现特定的类加载机制。

方法签名

ClassLoader 类中的 defineClass 方法有几个重载版本。最常用的版本之一的签名如下:

protected final Class<?> defineClass(String name, byte[] b, int off, int len)

参数说明:

  • name:预期的类名,可以为 null
  • b:包含类定义信息的字节数组。
  • off:类定义的起始偏移量(在字节数组中)。
  • len:类定义的长度。

使用场景

  • 自定义类加载器:当开发者需要从非标准源加载类(如网络资源、加密文件)时,会创建自定义类加载器,并在其中使用 defineClass 方法来实现类的加载。
  • 动态类加载:在一些动态生成类的应用场景中(如某些框架或工具),defineClass 方法用于将运行时生成的字节码转换为 Class 对象。

安全和限制

  • 安全检查defineClass 方法在将字节数组转换为 Class 对象时,会执行安全检查,确保类的结构和行为符合 JVM 规范。
  • 访问权限:由于这个方法被定义为 protected,因此只能在 ClassLoader 类或其子类中被调用。
  • 类名限制:如果指定了类名(name 参数非 null),那么加载的类名必须与此名称匹配。
  • 封装性限制:在 Java 9 及以后版本中,模块系统的引入对类加载器的行为做了进一步的限制,特别是在封装性方面。

示例

下面是一个简单的示例,展示如何在自定义类加载器中使用 defineClass

public class MyClassLoader extends ClassLoader {public Class<?> defineClass(String name, byte[] b) {return defineClass(name, b, 0, b.length);}
}

在这个示例中,MyClassLoader 类扩展了 ClassLoader,并提供了一个方法来定义类。这个方法接受一个类名和一个字节数组,然后调用 defineClass 来创建 Class 对象。

结论

defineClass 方法是 Java 类加载机制的一个重要组成部分,它使得开发者可以在运行时动态地加载和定义类。这个方法的正确使用对于实现灵活且安全的类加载策略至关重要。


总结

在 Java 中,当使用 ClassLoader 类的 loadClass 方法加载一个类时,loadClass 方法内部会按照一定的逻辑来决定如何加载这个类。如果这个类之前没有被加载过,loadClass 方法会最终调用 findClass 方法来加载这个类。这是 ClassLoader 类的内部机制,因此在使用 loadClass 方法时,不需要直接调用 findClass 方法。下面是这个过程的简化描述:

  1. 调用 loadClass 方法:当我们在 main 函数中调用 myClassLoader.loadClass("your_class_name") 时,实际上是调用了 ClassLoader 类的 loadClass 方法。

  2. 检查类是否已加载loadClass 方法首先检查这个类是否已经被加载过。如果已经加载,它会直接返回这个类的 Class 对象。

  3. 委托给父类加载器:如果类还没有被加载,loadClass 方法会尝试让父类加载器去加载这个类。如果父类加载器不存在或无法加载该类,loadClass 方法会调用 findClass 方法。

  4. 调用 findClass 方法findClass 方法是一个 protected 方法,通常在自定义类加载器中被重写。它包含了从特定来源(如文件系统、网络等)加载类的具体逻辑。

  5. 返回 Class 对象:一旦 findClass 方法成功加载了类,并返回了相应的 Class 对象,loadClass 方法就会将这个 Class 对象返回给调用者。

一般在自定义类加载器的实现中,MyClassLoader 类会重写 findClass 方法以从特定路径加载类文件。当我们在 main 方法中调用 loadClass 时,如果 your_class_name 类之前未被加载,MyClassLoaderloadClass 方法会间接调用我们重写的 findClass 方法来加载这个类。这就是为什么在 main 函数中并没有显式调用 findClass 方法,但该方法仍然被执行的原因。

如下是一个测试代码:

package per.mjn.t3;import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;public class Load7 {public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException {// 创建自定义类加载器对象MyClassLoader myClassLoader = new MyClassLoader();Class<?> c1 = myClassLoader.loadClass("Demo1_1");Class<?> c2 = myClassLoader.loadClass("Demo1_1");System.out.println(c1);// 判断两次加载获得的类对象是否相同System.out.println(c1 == c2); // true// 即使多次执行loadClass方法,但实际上类文件只会加载一次,第一次加载后就会放在自定义类加载器的缓存中// 下次再调用loadClass()时就可以在缓存中找到了,不会重复的进行类加载// -------------------------------------------------------MyClassLoader myClassLoader1 = new MyClassLoader();Class<?> c3 = myClassLoader1.loadClass("Demo1_1");// 唯一确定类的方式是,包名 类名相同,而且类加载器也是同一个,才认为这两个类才是完全一致的// 虽然MapImpl1与刚才的包名类名一样,但是由于他俩的类加载器对象不是同一个,所以认为这两个类不是同一个类// 也就是,这个类会被加载两次,因为是不同的类加载器,就认为他俩是相互隔离的,不会产生冲突System.out.println(c1 == c3); // false// 创建这个类的实例对象会触发静态代码块的执行c1.newInstance();  // 会打印出"init..."(在静态代码块中提前写好的)}
}class MyClassLoader extends ClassLoader {@Override   // name 就是类名称protected Class<?> findClass(String name) throws ClassNotFoundException {String path = "D:\\ideaProject\\JVM_Detect\\src\\per\\mjn\\t1\\" + name + ".class";ByteArrayOutputStream os = new ByteArrayOutputStream();try {Files.copy(Paths.get(path), os);// 得到字节数组byte[] bytes = os.toByteArray();// byte[] -> *.classClass<?> aClass = defineClass(name, bytes, 0, bytes.length);return aClass;} catch (IOException e) {e.printStackTrace();throw new ClassNotFoundException("类文件未找到", e);}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/186933.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

智跃人力资源管理系统 SQL注入漏洞复现

0x01 产品简介 智跃人力资源管理系统是基于B/S网页端广域网平台&#xff0c;一套考勤系统即可对全国各地多个分公司进行统一管控&#xff0c;成本更低。信息共享更快。跨平台&#xff0c;跨电子设备 0x02 漏洞概述 智跃人力资源管理系统GenerateEntityFromTable.aspx接口处存在…

SQL Sever 基础知识 - 数据查询

SQL Sever 基础知识 - 一、查询数据 一、查询数据第1节 基本 SQL Server 语句SELECT第2节 SELECT语句示例2.1 SELECT - 检索表示例的某些列2.2 SELECT - 检索表的所有列2.3 SELECT - 对结果集进行筛选2.4 SELECT - 对结果集进行排序2.5 SELECT - 对结果集进行分组2.5 SELECT - …

正则表达式及文本三剑客grep sed awk

正则表达式 1.元字符 . //匹配任意单个字符&#xff0c;可以是个汉字 [yang] //匹配范围内的任意单个字符 [^y] //匹配处理指定范围外的任意单个字符 [:alnum:] //字母和数字 [:alpha:] //代表…

Tomcat定时重启脚本

利用Linux系统定时任务重启tomcat&#xff0c;脚本文件参见附件。 系统定时任务设置方式参见《tomcat日志清理篇》。 tomcat定时重启&#xff1a; 1.先写重启脚本&#xff1a;&#xff08;参考附件&#xff09; 2. chmod 777 xxxx.sh 赋予权限 3. crontab -e 修改定时任务&am…

uc_12_进程间通信IPC_有名管道_无名管道

1 内存壁垒 进程间天然存在内存壁垒&#xff0c;无法通过交换虚拟地址直接进行数据交换&#xff1a; 每个进程的用户空间都是0~3G-1&#xff08;32位系统&#xff09;&#xff0c;但它们所对应的物理内存却是各自独立的。系统为每个进程的用户空间维护一张专属于该进程的内存映…

ZPLPrinter Emulator SDK for .NET 6.0.23.1123​ Crack

ZPLPrinter Emulator SDK for .NET 适用于 .NET 的 ZPLPrinter 仿真器 SDK 允许您通过编写 C# 或VB.NET 代码针对任何 .NET Framework、.NET CORE、旧版 ASP.NET MVC 和 CORE、Xamarin、Mono 和通用 Windows 平台 (UWP) 作业。 适用于 .NET 的 ZPLPrinter 仿真器 SDK 允许您将…

第一百八十五回 如何禁止页面跟随手机自动旋转

文章目录 1. 概念介绍2. 使用方法2.1 全面禁止2.2 局部禁止3. 示例代码4. 内容总结我们在上一章回中介绍了"如何自定义Radio组件"相关的内容,本章回中将介绍 如何禁止页面随手机自动旋转.闲话休提,让我们一起Talk Flutter吧。 1. 概念介绍 在手机默认设置下,手机…

数据爬虫(JSON格式)数据地图可视化(pyecharts)【步骤清晰,一看就懂】

一、前言 数据存储在网页上&#xff0c;需要爬取数据下来&#xff0c;数据存储格式是JSON&#xff0c;数据可视化在工作中也变得日益重要&#xff0c;接下来将数据爬虫与数据可视化结合起来&#xff0c;做个案例 注&#xff1a;当时数据是22年1月29日爬取数据 二、使用步骤 …

直播前期准备

直播前的准备是一个综合性的过程&#xff0c;需要从多个方面进行考虑和准备。以下是一些直播前准备的参考∶ 1.确定直播主题和目标∶明确直播的主题和目标&#xff0c;以及如何吸引观众。考虑观众的兴趣和需求&#xff0c;选择一个熟悉且具有吸引力的主题&#xff0c;以提升直…

js事件流与事件委托/事件代理

1 事件流 事件流分为两步&#xff0c;一是捕获&#xff0c;二是冒泡 1.1 捕获概念 捕获就是从最高层一层一层往下找到最内部的节点 1.2 冒泡概念 捕获到最小节点后&#xff0c;一层一层往上返回&#xff0c;像是气泡从最底部往上冒一样&#xff0c;由于水深不同压强不同&…

C题目12:请写一个函数,判断一个数是否为质数,并在main函数中调用

一.每日小语 人的一切痛苦&#xff0c;本质上都是对自己的无能的愤怒。——王小波 自己思考 判断一个函数是否为质数&#xff0c;这个我在之前练过&#xff0c;我想至少两次&#xff0c;而这一次则是问我如何在main函数中调用&#xff0c;这个概念我不理解&#xff0c;所以我…

HarmonyOS4.0 ArkUI组件

目录 简介 搭建开发环境 ArkUI基础组件 Image组件 Text组件 TextInput Button Slider 简介 HarmonyOS 4.0的ArkUI组件是一套UI开发框架&#xff0c;提供开发者进行应用UI开发时所必须的能力。在ArkUI中&#xff0c;组件是界面搭建与显示的最小单位&#xff0c;开发者通过…

linux拨号服务器如何创建爬虫ip池

作为一个爬虫技术员&#xff0c;除了要熟练掌握至少一种编程语言外&#xff0c;还应该创建属于自己的爬虫ip池。我们都知道&#xff0c;在进行爬虫采集时&#xff0c;经常会遇到网站各种发爬机制&#xff0c;如果有自己的ip池&#xff0c;将会让爬虫这项枯燥无味的工作变得非常…

OLED双面显示广告机的应用场景

OLED双面显示广告机是一种创新的广告设备&#xff0c;它具有双面显示屏幕&#xff0c;可以同时向两个方向展示广告或信息。这种设备被广泛应用于各种场景&#xff0c;例如&#xff1a; 商业展示&#xff1a;在大型商业场所&#xff0c;如购物中心、百货商场等&#xff0c;OLED双…

RPC 简介

RPC 简介 RPC代表远程过程调用&#xff08;Remote Procedure Call&#xff09;&#xff0c;它是一种通信协议&#xff0c;允许一个计算机程序调用另一个地址空间&#xff08;通常是另一台机器上&#xff09;的过程或函数&#xff0c;就像本地调用一样&#xff0c;而无需显式编…

Git常用命令#切换分支

要在 Git 中切换分支&#xff0c;你可以使用 git checkout 命令。 a.创建新分支并切换到该分支 如果你想要创建一个新分支并立即切换到该分支&#xff0c;可以使用以下命令&#xff1a; git checkout -b 新分支名这会创建一个名为 新分支名 的新分支&#xff0c;并将你的工作目…

运维知识点-Nginx

Nginx Nginx解析安全实战预备知识实验目的#制作图片木马# web服务器-Nginx服务命令及配置centOS7安装安装所需插件安装gccpcre、pcre-devel安装zlib安装安装openssl Nginx解析安全实战 预备知识 NginxPHP/FastCGI构建的WEB服务器工作原理 Nginx|FastCGI简介 Nginx (“engin…

飞天使-elk搭建补充

文章目录 es 集群创建密码kibana 配置文件以及和nginx配置pm2 安装定期清理索引以及告警logstash 配置filebeat 配置文件nginx 的日志索引 es 集群创建密码 参考这篇博文进行设置&#xff1a;https://juejin.cn/post/7079955586330132487 最后的效果 #curl -XGET http://127.0…

微服务详细介绍(什么是微服务)

微服务是一种架构风格&#xff0c;它将复杂的应用程序拆分成一系列小型、独立的服务。每个服务都可以独立部署、扩展和维护&#xff0c;而无需对整个应用程序进行修改。 微服务架构有以下特点&#xff1a; 服务拆分&#xff1a;将应用程序拆分成多个小型服务。 独立部署&…

云轴科技ZStack信创云平台助力国泰君安期货实现信创改造

信创是数字中国建设的重要组成部分&#xff0c;也是数字经济发展的关键推动力量。作为云基础软件企业&#xff0c;云轴科技ZStack 产品矩阵全面覆盖数据中心云基础设施&#xff0c;ZStack信创云首批通过可信云《一云多芯IaaS平台能力要求》先进级&#xff0c;是其中唯一兼容四种…