返回顶部
首页 > 资讯 > 精选 >Java的ClassLoader有什么用
  • 373
分享到

Java的ClassLoader有什么用

2023-06-17 13:06:58 373人浏览 薄情痞子
摘要

这篇文章主要介绍“Java的ClassLoader有什么用”,在日常操作中,相信很多人在Java的ClassLoader有什么用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java的ClassLoader有

这篇文章主要介绍“Java的ClassLoader有什么用”,在日常操作中,相信很多人在Java的ClassLoader有什么用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java的ClassLoader有什么用”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

ClassLoader

类加载器(class loader)用来加载 Java 类到 Java 虚拟机中。Java 源程序(.java 文件)在经过 Java 编译器编译之后就被转换成 Java 字节代码(.class 文件)。类加载器负责读取 Java 字节代码,并转换成 java.lang.Class 类的一个实例。

ClassLoader Hierarchy

JVM在加载类时,使用的是双亲委托模式(delegation model),也就是说除了Bootstrap ClassLoader之外,每个ClassLoader都有一个Parent ClassLoader。ClassLoader是按需进行加载class文件。当ClassLoader试图加载一个类时,首先检查本地缓冲,查看类是否已被加载,如果类没有被加载,尝试委托给父ClassLoader进行加载,如果父ClassLoader加载失败,才会由该ClassLoader进行加载,从而避免了重复加载的问题。一下为类装载器层次图:

Java的ClassLoader有什么用

Bootstrap ClassLoader:负责加载java_home/lib目录下的核心类或- Xbootclasspath指定目录下的类。

Extension ClassLoader:负责加载java_home/lib/ext目录下的扩展类或 -Djava.ext.dirs 指定目录下的类。

System ClassLoader:负责加载-classpath/-Djava.class.path所指的目录下的类。

如果类App1在本地缓冲中没有class文件(没有被加载),那么它会自底向上依次查找是否已经加载了类,如果已经加载,则直接返回该类实例的引用。如果BootstrapClassLoader也未成功加载该类,那么会抛出异常,然后自顶向下依次尝试加载,如果到App1 ClassLoader还没有加载成功,那么会抛出ClassNotFoundException异常给调用者。

Java代码

public static void main(String[] args) {             ClassLoader cl = ClassLoader.getSystemClassLoader();             while(cl != null){                 System.out.println(cl);                 System.out.println("parent class loader: " + cl.getParent());                 cl = cl.getParent();             }         }

Java代码

sun.misc.Launcher$AppClassLoader@19821f    parent class loader: sun.misc.Launcher$ExtClassLoader@addbf1    sun.misc.Launcher$ExtClassLoader@addbf1    parent class loader: null

我们看到,当前系统类装载器为AppClassLoader,AppClassLoader的父类装载器是ExtClassLoader,ExtClassLoader的父装载器为null,表示为BootstrapClassLoader。BootstrapClassLoader由JVM采用本地代码实现,因此没有对应的Java类,所以ExtClassLoader的getParent()返回null。

ClassLoader的职责之一是保护系统名字空间。以下为ClassLoader类部分代码:

Java代码

private ProtectionDomain preDefineClass(String name,                             ProtectionDomain protectionDomain)         {         if (!checkName(name))             throw new NoClassDefFoundError("IllegalName: " + name);             if ((name != null) && name.startsWith("java.")) {             throw new SecurityException("Prohibited package name: " +                         name.substring(0, name.lastIndexOf('.')));         }         if (protectionDomain == null) {             protectionDomain = getDefaultDomain();         }             if (name != null)             checkCerts(name, protectionDomain.getCodeSource());             return protectionDomain;         }

那么,当我们定义如下类Foo,虽然能够通过编译,但是会报java.lang.SecurityException: Prohibited package name: java.lang异常,因为我们试图将Foo类写入到java.lang包下。

Java代码

package java.lang;         public class Foo {                  public static void main(String args[]) throws Exception {             Foo f = new Foo();             System.out.println(f.toString());         }     }

定制ClassLoader

Java自带的ClassLoader类的定义为:

Java代码

public abstract class ClassLoader{      }

启动类加载器是JVM通过调用ClassLoader.loadClass()方法。

Java代码

public Class loadClass(String name) throws ClassNotFoundException {         return loadClass(name, false);         }         protected synchronized Class loadClass(String name, boolean resolve)         throws ClassNotFoundException         {         // First, check if the class has already been loaded         Class c = findLoadedClass(name);         if (c == null) {             try {             if (parent != null) {                 c = parent.loadClass(name, false);             } else {                 c = findBootstrapClass0(name);             }             } catch (ClassNotFoundException e) {                 // If still not found, then invoke findClass in order                 // to find the class.                 c = findClass(name);             }         }         if (resolve) {             resolveClass(c);         }         return c;         }         protected Class findClass(String name) throws ClassNotFoundException {         throw new ClassNotFoundException(name);         }

loadClass(String name, boolean resolve)方法中的resolve如果为true,表示分析这个Class对象,包括检查Class Loader是否已经初始化等。loadClass(String name) 在加载类之后不会对该类进行初始化,直到***次使用该类时,才会对该类进行初始化。

那么,我们在定制ClassLoader的时候,通常只需要覆写findClass(String name)方法。在findClass(String name)方法内,我们可以通过文件、网络(URL)等形式获取字节码。以下为获取字节码的方法:

Java代码

public InputStream getResourceAsStream(String name);     public URL getResource(String name);     public InputStream getResourceAsStream(String name);     public Enumeration getResources(String name) throws IOException;

在取得字节码后,需要调用defineClass()方法将字节数组转换成Class对象,该方法签名如下:

Java代码

protected final Class defineClass(String name, byte[] b, int off, int len,                          ProtectionDomain protectionDomain)         throws ClassFORMatError

对于相同的类,JVM最多会载入一次。如果同一个class文件被不同的ClassLoader载入(定义),那么载入后的两个类是完全不同的。

Java代码

public class Foo{         //         private static final AtomicInteger COUNTER = new AtomicInteger(0);             public Foo() {             System.out.println("counter: " + COUNTER.incrementAndGet());         }                  public static void main(String args[]) throws Exception {             URL urls[] = new URL[]{new URL("file:/c:/")};             URLClassLoader ucl1 = new URLClassLoader(urls);             URLClassLoader ucl2 = new URLClassLoader(urls);             Class c1 = ucl1.loadClass("Foo");             Class c2 = ucl2.loadClass("Foo");             System.out.println(c1 == c2);             c1.newInstance();             c2.newInstance();         }     }

以上程序需要保证Foo.class文件不在classpath路径下。从而使AppClassLoader无法加载Foo.class。

输出结果:

Java代码

false    counter: 1    counter: 1

WEB应用的ClassLoader

绝大多数的EJB容器,Servlet容器等都会提供定制的ClassLoader,来实现特定的功能。但是通常情况下,所有的servlet和filter使用一个ClassLoader。每个jsp都使用一个独立的ClassLoader。

隐式(implicit)和显示(explicit)的加载

隐式加载:我们使用new关键字实例化一个类,就是隐身的加载了类。

显示加载分为两种:

java.lang.Class的forName()方法;

java.lang.ClassLoader的loadClass()方法。

Class.forName()方法有两个重载的版本:

Java代码

public static Class forName(String className)                      throws ClassNotFoundException {             return forName0(className, true, ClassLoader.getCallerClassLoader());         }         public static Class forName(String name, boolean initialize,                        ClassLoader loader)             throws ClassNotFoundException

可以看出,forName(String className)默认以true和ClassLoader.getCallerClassLoader()调用了三参数的重载方法。ClassLoader.getCallerClassLoader()表示以caller class loader加载类,并会初始化类(即静态变量会被初始化,静态初始化块中的代码也会被执行)。如果以false和ClassLoader.getCallerClassLoader()调用三参数的重载方法,表示加载后的类不会被初始化。

ClassLoader.loadClass()方法在类加载后,也同样不会初始化类。

两个异常(exception)

NoClassDefFoundError: 当java源文件已编译成.class文件,但是ClassLoader在运行期间搜寻路径load某个类时,没有找到.class文件则抛出这个异常。

ClassNotFoundException: 试图通过一个String变量来创建一个Class类时不成功则抛出这个异常

到此,关于“Java的ClassLoader有什么用”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

--结束END--

本文标题: Java的ClassLoader有什么用

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

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

猜你喜欢
  • Java的ClassLoader有什么用
    这篇文章主要介绍“Java的ClassLoader有什么用”,在日常操作中,相信很多人在Java的ClassLoader有什么用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java的ClassLoader有...
    99+
    2023-06-17
  • Java的ClassLoader是什么
    本文小编为大家详细介绍“Java的ClassLoader是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java的ClassLoader是什么”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。ClassLoad...
    99+
    2023-06-16
  • java classloader的使用方法是什么
    Java ClassLoader是Java虚拟机(JVM)的一个重要组成部分,用于加载Java类文件。ClassLoader负责将编...
    99+
    2024-04-09
    java
  • Java的ClassLoader机制是什么
    本篇内容介绍了“Java的ClassLoader机制是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!JVM在加载类的时候,都是通过Cla...
    99+
    2023-06-17
  • java classloader的工作机制是什么
    Java的ClassLoader是Java虚拟机(JVM)的一个重要组成部分,它主要负责加载Java类文件并把字节码文件转换成运行时...
    99+
    2024-04-09
    java
  • JAVA的classloader怎么编写
    今天小编给大家分享一下JAVA的classloader怎么编写的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。什么是 Clas...
    99+
    2023-06-03
  • JAVA classloader怎么理解
    这篇文章给大家介绍JAVA classloader怎么理解,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。什么是 ClassLoader在流行的商业化编程语言中,Java 语言由于在 Java 虚拟机 (JVM) 上运行...
    99+
    2023-06-03
  • Java中的ClassLoader核心知识点有哪些
    这篇文章主要介绍“Java中的ClassLoader核心知识点有哪些”,在日常操作中,相信很多人在Java中的ClassLoader核心知识点有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java中的C...
    99+
    2023-06-16
  • Java类加载器ClassLoader的使用详解
    目录BootstrapClassLoaderExtClassLoaderAppClassLoader类加载器的具体实现在哪里类加载器的初始化时机如何进行的类加载Loader.getR...
    99+
    2022-12-19
    Java类加载器ClassLoader Java ClassLoader Java类加载器
  • Java中ClassLoader类加载的示例分析
    这篇文章主要为大家展示了“Java中ClassLoader类加载的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Java中ClassLoader类加载的示例分析”这篇文章吧。双亲委派模型...
    99+
    2023-05-30
    java classloader
  • java的类有什么用?
    java的类有什么用?类是具有共同属性和行为的对象集合,类定义了对象的属性和方法。可以说,类就是一个模板,定义多个对象共同的属性和方法。通过类可以实例化多个该类的对象,每个对象的属性值不同,可以说类好比设计图纸,而对象好比根据图纸建设出来的...
    99+
    2017-12-21
    java教程 java
  • Java的ScheduledThreadPoolExecutor有什么用
    Java的ScheduledThreadPoolExecutor是一个实现了ScheduledExecutorService接口的线...
    99+
    2023-09-09
    Java
  • 如何在java中利用ClassLoader加载指定的class文件
    这篇文章给大家介绍如何在java中利用ClassLoader加载指定的class文件,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。首先定义一个类,比如MyTest,并且将其编译成class文件,然后放到一个指定的文件夹...
    99+
    2023-06-06
  • java的接口有什么用?
    Java接口是一系列方法的声明,是一些方法特征的集合,一个接口只有方法的特征没有方法的实现,因此这些方法可以在不同的地方被不同的类实现,而这些实现可以具有不同的行为(功能)。接口作用如下:丰富Java面向对象的思想:在Java语言中, ab...
    99+
    2016-05-26
    java教程 java 接口
  • java中的session.setattribute有什么用
    在Java中,session.setAttribute()方法用于在Session中存储一个对象。它接受两个参数,第一个参数是一个字...
    99+
    2024-02-29
    java
  • java的random类有什么用
    Random类是Java中的一个随机数生成器类,它可以用于生成种子为不确定的伪随机数。它有以下几个主要的用途: 生成随机数:通过...
    99+
    2024-02-29
    java
  • java类有什么用
    java中的类是构造面向对象程序的基本单位,是抽取了同类对象的共同属性和方法所形成的对象或实体的“模板”。Java中的对象 (推荐学习:java课程)现在让我们深入了解什么是对象。看看周围真实的世界,会发现身边有很多对象...
    99+
    2020-03-13
    java入门 java
  • java redis有什么用
    Java连接redis的使用示例Redis是开源的key-value存储工具,redis通常用来存储结构化的数据,因为redis的key可以包含String、hash、listset和sorted list。Redisclient支持多种语...
    99+
    2019-03-10
    java教程 java redis
  • java有什么用呢
    java的用处:1.编写桌面应用程序;2.编写Web应用程序;3.编写分布式系统和嵌入式系统应用程序等等;可用于编写桌面应用程序、Web应用程序、分布式系统和嵌入式系统应用程序等,例如:许多的 Android应用都是Java程序员...
    99+
    2024-04-02
  • Java java.time.Instant.isBefore()有什么用
    Java java.time.Instant.isBefore()方法用于比较当前Instant对象是否在指定的Instant对象之...
    99+
    2023-10-24
    Java
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作