返回顶部
首页 > 资讯 > 精选 >怎么在java中自定义一个类加载器
  • 312
分享到

怎么在java中自定义一个类加载器

java 2023-05-30 18:05:01 312人浏览 八月长安
摘要

本篇文章为大家展示了怎么在java中自定义一个类加载器,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。如果要使用自定义类加载器加载class文件,就需要继承java.lang.ClassLoader类

本篇文章为大家展示了怎么在java中自定义一个类加载器,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。

如果要使用自定义类加载器加载class文件,就需要继承java.lang.ClassLoader类。

ClassLoader有几个重要的方法:

protectedClassLoader(ClassLoaderparent):使用指定的、用于委托操作的父类加载器创建新的类加载器。

protectedfinalClass<?>defineClass(Stringname,byte[]b,intoff,intlen):将一个byte数组转换为Class类的实例。

protectedClass<?>findClass(Stringname):使用指定的二进制名称查找类。

publicClass<?>loadClass(Stringname):使用指定的二进制名称来加载类。

protectedfinalClass<?>findLoadedClass(Stringname):如果Java虚拟机已将此加载器记录为具有给定二进制名称的某个类的启动加载器,则返回该二进制名称的类。否则,返回null。

publicfinalClassLoadergetParent():返回委托的父类加载器。

protectedfinalvoidresolveClass(Class<?>c):链接指定的类。

如果要遵循双亲委派模型,则重写findClass(Stringname)方法;如果不想遵循双亲委派模型,则直接重写loadClass(Stringname)方法。

自定义遵循双亲委派模型的类加载器

ParentsDelegateClassLoader.java

package com.zzj.classloader;import java.io.ByteArrayOutputStream;import java.io.File;import java.io.FileInputStream;import java.io.IOException;import java.io.InputStream;public class ParentsDelegateClassLoader extends ClassLoader {private static final String EXT = ".class";private String path;public ParentsDelegateClassLoader() {path = this.getResource("").getPath();}public ParentsDelegateClassLoader(String path) {this.path = path;}@Override   protected Class<?> findClass(String name) throws ClassNotFoundException {byte[] b = null;try {b = loadClassFile(name);}catch (IOException e) {e.printStackTrace();}return this.defineClass(name, b, 0, b.length);}private byte[] loadClassFile(String name) throws IOException {String classFile = getClassFile(name);System.out.println("即将加载class文件" + classFile);ByteArrayOutputStream out = new ByteArrayOutputStream();InputStream input = new FileInputStream(classFile);int count;byte[] temp = new byte[1024];while ((count = input.read(temp)) > -1) {out.write(temp, 0, count);}out.close();input.close();return out.toByteArray();}private String getClassFile(String name) {String pathName = name.replace(".", File.separator);if (path.endsWith("/") || path.endsWith("\\")) {return path + pathName + EXT;}return path + File.separator + pathName + EXT;}}

现在类路径classpath下和F:\\ClassloaderTest\\bin目录下都有一个类文件com\zzj\classloader\User.class,包名为com.zzj.classloader,使用类加载器ParentsDelegateClassLoader加载F:\\ClassloaderTest\\bin下的类。

package com.zzj.classloader;public class App {private static final String path = "F:\\ClassloaderTest\\bin";private static final String classname = "com.zzj.classloader.User";public static void main(String[] args) throws Exception {ParentsDelegateClassLoader classLoader = new ParentsDelegateClassLoader(path);Class<?> clazz = classLoader.loadClass(classname);System.out.println(clazz);System.out.println(clazz.getClassLoader());}}

输出:

class com.zzj.classloader.User sun.misc.Launcher$AppClassLoader@19821f

User类的加载器是系统类加载器AppClassLoader,而不是我们自己定义的类加载。实际上被加载不是F:\\ClassloaderTest\\bin下的类,而是classpath下的类。这就是双亲委派模型:当ParentsDelegateClassLoader加载器接收到加载请求后,会先委托给父类加载器,如果父类加载器加载成功,则返回一个Class对象。如果加载失败,才会让接收到加载请求的类加载器加载。

把classpath下的User类删掉测试运行:

即将加载class文件F:\ClassloaderTest\bin\com\zzj\classloader\User.class class com.zzj.classloader.User com.zzj.classloader.ParentsDelegateClassLoader@61de33

此时User类的加载为ParentsDelegateClassLoader。

这一点可以从ClassLoader的源码中得到验证:

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

调用了重载方法:

protected synchronized Class<?> loadClass(String name, boolean resolve)   throws ClassNotFoundException   {   // 先判断该类是否已被当前的类加载器加载   Class c = findLoadedClass(name);   if (c == null) {     try {     if (parent != null) {// 如果存在父类加载器,则委派给父类加载       c = parent.loadClass(name, false);     } else {// 如果父类加载为空,则其父类加载器为引导类加载器       c = findBootstrapClass0(name);     }     } catch (ClassNotFoundException e) {       // 如果父类加载器加载失败,则自己加载,调用的就是findClass方法!       c = findClass(name);     }   }   if (resolve) {     resolveClass(c);   }   return c;   }

可见,如果想要破坏双亲委派模型,可以直接重写loadClass(Stringname)方法。

自定义不遵循双亲委派模型的类加载器

NotParentsDelegateClassLoader.java

package com.zzj.classloader;import java.io.ByteArrayOutputStream;import java.io.File;import java.io.FileInputStream;import java.io.FileNotFoundException;import java.io.IOException;import java.io.InputStream;public class NotParentsDelegateClassLoader extends ClassLoader {private static final String EXT = ".class";private String path;public NotParentsDelegateClassLoader() {path = this.getResource("").getPath();}public NotParentsDelegateClassLoader(String path) {this.path = path;}@Override   public Class<?> loadClass(String name) throws ClassNotFoundException {byte[] b = null;try {b = loadClassFile(name);}catch (FileNotFoundException e) {System.err.println("加载器" + this.getClass().getName()           + "没有找到class文件" + name + ",将委派给父类加载器!");// 委派给父类加载器 return getClass().getClassLoader().loadClass(name);}catch (IOException e) {System.err.println("加载器" + this.getClass().getName() + "加载class文件"           + name + "失败,将委派给父类加载器!");// 委派给父类加载器 return getClass().getClassLoader().loadClass(name);}// 检查该类是否被当前类加载器加载过(只检查当前类加载器,不会检查父类加载器) Class<?> clazz = findLoadedClass(name);if (clazz != null) {System.out.println("类" + name + "已被加载过!");return clazz;} else {System.out.println("类" + name + "尚未被加载!");}return this.defineClass(name, b, 0, b.length);}private byte[] loadClassFile(String name) throws IOException,       FileNotFoundException {String classFile = getClassFile(name);System.out.println("即将加载class文件" + classFile);ByteArrayOutputStream out = new ByteArrayOutputStream();InputStream input = new FileInputStream(classFile);int count;byte[] temp = new byte[1024];while ((count = input.read(temp)) > -1) {out.write(temp, 0, count);}out.close();input.close();return out.toByteArray();}private String getClassFile(String name) {String pathName = name.replace(".", File.separator);if (path.endsWith("/") || path.endsWith("\\")) {return path + pathName + EXT;}return path + File.separator + pathName + EXT;}}

现在类路径classpath下有一个类文件com\zzj\classloader\User.class,包名为com.zzj.classloader,使用类加载器NotParentsDelegateClassLoader加载User类。

package com.zzj.classloader;public class App2 {private static final String className = "com.zzj.classloader.User";public static void main(String[] args) throws Exception {NotParentsDelegateClassLoader classLoader = new NotParentsDelegateClassLoader();Class<?> clazz = classLoader.loadClass(className);System.out.println(clazz);System.out.println(clazz.getClassLoader());}}

输出:

即将加载class文件/E:/Myeclipse/zzjtest/WEBRoot/WEB-INF/classes/com\zzj\classloader\User.class 类com.zzj.classloader.User尚未被加载! 即将加载class文件/E:/Myeclipse/zzjtest/WebRoot/WEB-INF/classes/java\lang\Object.class class com.zzj.classloader.User com.zzj.classloader.NotParentsDelegateClassLoader@61de33 加载器com.zzj.classloader.NotParentsDelegateClassLoader没有找到class文件java.lang.Object,将委派给父类加载器!

此时User类的加载器是NotParentsDelegateClassLoader,没有先委托给父类,只有加载失败才会委托给父类加载器,正好跟双亲委派模型是反的。

当然,即使加载失败,也可以不委托给父类加载器,而指定其他的类加载器,从而可以构建更加复杂的网状模型的类加载机制

上述内容就是怎么在java中自定义一个类加载器,你们学到知识或技能了吗?如果还想学到更多技能或者丰富自己的知识储备,欢迎关注编程网精选频道。

--结束END--

本文标题: 怎么在java中自定义一个类加载器

本文链接: https://lsjlt.com/news/220474.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 怎么在java中自定义一个类加载器
    本篇文章为大家展示了怎么在java中自定义一个类加载器,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。如果要使用自定义类加载器加载class文件,就需要继承java.lang.ClassLoader类...
    99+
    2023-05-30
    java
  • Java中如何自定义一个类加载器
    目录如何自定义加载器?示例:读取某文件的下的某class文件类加载器的使用及自定义类加载器如何自定义加载器? 1.创建一个自定义加载器类 继承 ClassLoader 类 2.重写 ...
    99+
    2022-11-21
    Java自定义类加载器 自定义类加载器 Java类加载器
  • java怎么自定义类加载器
    要自定义类加载器,可以按照以下步骤进行:1. 创建一个类,继承自java.lang.ClassLoader类,覆盖其中的findCl...
    99+
    2023-09-14
    java
  • java类加载器怎么自定义
    要自定义Java类加载器,可以继承ClassLoader类,并重写它的findClass()方法。下面是一个简单的示例:```jav...
    99+
    2023-09-14
    java
  • Java怎么自定义类加载器实现类隔离
    这篇“Java怎么自定义类加载器实现类隔离”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Java怎么自定义类加载器实现类隔离...
    99+
    2023-07-05
  • Java基础之自定义类加载器
    目录一、类加载器关系二、基于本地class文件的自定义类加载器三、遇到的问题四、基于网络(url)class文件的自定义类加载器一、类加载器关系 自定义类加载器 创建一个类继承C...
    99+
    2024-04-02
  • 如何在Java中自定义一个枚举类
    本文章向大家介绍如何在Java中自定义一个枚举类的基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。Java的特点有哪些Java的特点有哪些1.Java语言作为静态面向对象编程语言的代表,实现了面向对象理论,允许程序...
    99+
    2023-06-06
  • 在c#中怎么定义一个类
    在 c# 中,通过使用 class 关键字定义一个类,它是一个模板,用于创建具有特定行为和属性的对象。具体步骤包括:使用 class 关键字定义类名,定义字段、属性、构造函数、方法和事件...
    99+
    2024-05-12
    c#
  • Java自定义类加载器实现类隔离详解
    目录一、背景二、解决思路2.1 创建依赖所在目录2.2 定义操作接口2.3 编写并构建业务包2.4 自定义类加载器2.5 主流程步骤三、总结一、背景 某服务需要连接操作多种组件(每种...
    99+
    2023-03-02
    Java类加载器实现类隔离 Java类加载器 类隔离 Java类加载器 Java 类隔离
  • java自定义类加载器如何实现类隔离
    目录自定义类加载器准备通过URLClassLoader来实现【推荐】通过继承ClassLoader实现网上java自定义类加载器很多容易找到,但是都是加载的单个类,如果被加载的类,有...
    99+
    2022-11-21
    java类隔离 自定义类加载器 java类加载器
  • Java通过自定义类加载器实现类隔离
    目录前言类隔离是什么使用场景解决方案重写findClass重写loadClass总结前言 由于微服务的快速迭代、持续集成等特性,越来越多的团队更倾向于它。但是也体现出了一些问题,比如...
    99+
    2022-11-13
    Java 类加载器 类隔离 Java 类加载器 Java 类隔离
  • 怎么在java中自定义异常类
    本篇文章给大家分享的是有关怎么在java中自定义异常类,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。Java可以用来干什么Java主要应用于:1. web开发;2. Andro...
    99+
    2023-06-14
  • 如何在Java中自定义封装一个JDBC工具类
    如何在Java中自定义封装一个JDBC工具类?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。步骤① 创建配置文件(config.properties),用于存放注册驱动和连接...
    99+
    2023-06-06
  • 怎么在java中定义一个变量
    怎么在java中定义一个变量?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。Java可以用来干什么Java主要应用于:1. web开发;2. Android开发;...
    99+
    2023-06-14
  • 在c#中怎么定义一个类数组
    类数组是用于存储相同类型对象的容器,在 c# 中可以使用以下语法进行定义:类名[] 数组名。在定义后,可以通过在声明时或声明后初始化数组。数组中的元素可以通过索引器或 for 循环进行访...
    99+
    2024-05-14
    c#
  • 在c#中怎么定义一个类变量
    在 c# 中,使用“[访问修饰符] static 数据类型 变量名;”定义类变量,例如 “public static int age;”。类变量具有全局作用域,可在类内任何位置访问,在所...
    99+
    2024-05-14
    c# 作用域
  • Java自定义类加载器及JVM自带的类加载器之间的交互关系是什么
    这篇“Java自定义类加载器及JVM自带的类加载器之间的交互关系是什么”文章,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要参考一下,对于“Java自定义类加载器及JVM自带的类加载器之间的交互关系是什么”,小编整理...
    99+
    2023-06-06
  • 怎么在Android中自定义一个控件
    怎么在Android中自定义一个控件?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。代码class SleepDayChart(context: Contex...
    99+
    2023-06-14
  • 怎么在Java中利用AQS实现一个自定义同步器
    这篇文章主要为大家详细介绍了怎么在Java中利用AQS实现一个自定义同步器,文中示例代码介绍的非常详细,具有一定的参考价值,发现的小伙伴们可以参考一下:Java的特点有哪些Java的特点有哪些1.Java语言作为静态面向对象编程语言的代表,...
    99+
    2023-06-06
  • jvm中怎么加载一个类
    这篇文章将为大家详细讲解有关jvm中怎么加载一个类,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。  第一步:加载,双亲委派:启动类加载器(jre/lib),系统扩展类加载器(ext/lib)...
    99+
    2023-06-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作