详解JAVA类加载器
创始人
2024-05-29 22:39:49
0

目录

1.概述

2.双亲委派

3.ServiceClassLoader

4.URLClassLoader

5.加载冲突


1.概述

概念:

类加载器(Class Loader)是Java虚拟机(JVM)的一个重要组件,负责加载Java类到内存中并使其可以被JVM执行。类加载器是Java程序的核心机制之一。

分类:

类加载器一共有三种:

  • 启动类加载器,加载系统类rt.jar。
  • 扩展类加载器,加载JDK内部,rt.jar之外,由于JDK版本迭代而新出现的扩展类。
  • 应用类加载器加载J用户所配置的classpath下的类。

三种类加载器之间从上到下有父子关系,上层是下层的父加载器。

2.双亲委派

在类加载中可能会遇见这样一种情况:

JDK里自带一个java.lang.String,有一个恶意的攻击类也将类的全类名命名为java.lang.String。如果我们想使用String类型的时候,类加载器将这个伪装成JDK的String类的恶意类加载执行,系统就将遭受到破坏。

很明显,为了安全起见JDK需要一种合理的加载方式来控制类的加载,这种合理的加载方式就是——双亲委派机制。

双亲委派机制总结起来就是:

先找父级加载器要,要是父级加载器没有,再由自己加载。

因为启动类加载器、扩展类加载器工作在应用的启动阶段,加载的也是JDK内核相关的jar,与应用没有太大关系,与应用相关的只有应用程序加载类,而应用程序的父级最多也就是扩展类加载器、启动类加载器,也就是最多找两个双亲拿,所以称为双亲委派机制。

整个类加载的详细流程为:

我们来看看任意一个ClassLoader的loadClass源码,整个源码中双亲委派的体现很直接:

类加载器在需要加载一个类(class)的时候会先判断类是否被自己加载过,

如果没有,调用其成员变量classloader parent委托其父载器帮忙加载,

父加载器再重复以上过程,如果递归到最顶级的启动类加载器,发现都加载不了,

当前启动器会自行加载。

3.ServiceClassLoader

SPI机制:

SPI(Service Provider Interface),是在JDK1.6中提出的一种基于接口的服务发现机制,它允许第三方服务提供者扩展框架的功能。在SPI机制中,框架定义一组接口,并规定这些接口的实现类必须以一定的命名规则放在特定的路径下,然后通过Java自带的SPI机制,动态地加载和实例化这些实现类。

基于SPI机制可以实现这样的生态结构,由官方制定一些标准,厂商去给出实现,也就是官方给出接口,由各大厂商去实现接口,如JAVA EE规范中的JDBC规范等等,支撑SPI实现的核心是一个类加载器——ServiceClassLoader。

ServiceClassLoader是一种用于加载和实例化服务提供者的工具类。服务提供者是指实现了特定接口或抽象类的类,而服务加载器则负责在运行时动态地查找和加载这些实现类。

  • 定义服务接口:定义一个Java接口或抽象类,用于描述服务提供者必须实现的行为。

  • 实现服务提供者:定义一个或多个服务提供者,实现服务接口或抽象类。

  • 配置服务提供者:在META-INF/services目录下创建一个以服务接口或抽象类名为文件名的文件,并在文件中列出所有实现服务接口或抽象类的类名。

  • 加载服务提供者:使用Service Loader类动态加载和实例化服务提供者。

代码示例:

接口:

package com.eryi;public interface HelloService {void sayHello();
}

实现类:

package com.eryi;public class ChineseHelloService implements HelloService{@Overridepublic void sayHello() {System.out.println("你好!");}
}
package com.eryi;public class EnglishHelloService implements HelloService{@Overridepublic void sayHello() {System.out.println("Hello!");}
}

发布服务:

在META-INF/services目录下创建一个名为"com.eryi.HelloService"的文件,文件内容如下:

com.eryi.EnglishHelloService
com.eryi.ChineseHelloService

服务发现:

package com.eryi;import java.util.ServiceLoader;public class test {public static void main(String[] args) {ServiceLoader loader = ServiceLoader.load(HelloService.class);for (HelloService service : loader) {service.sayHello();}}
}

运行结果:

4.URLClassLoader

启动类加载器、扩展类加载器、应用类加载器均是在程序工作之前完成对各个路径下所有jar包的加载的,这些路径里的jar包后续如果有什么变化,这三个类加载器是无法进行加载从而动态进行更新的。URLClassLoader就是为了实现这种动态加载而出现的。

URLClassLoader支持三种路径:

  • 目录

  • jar包

  • 网络

代码示例:

public class test {public static void main(String[] args) throws Exception {// 创建一个URLClassLoader,指定加载路径为当前目录URLClassLoader classLoader = new URLClassLoader(new URL[] { new URL("file:./") });// 动态加载HelloWorld类Class helloWorldClass = classLoader.loadClass("HelloWorld");// 创建HelloWorld对象Object helloWorld = helloWorldClass.newInstance();// 调用HelloWorld对象的sayHello方法helloWorldClass.getMethod("sayHello").invoke(helloWorld);}
}class HelloWorld {public void sayHello() {System.out.println("Hello, world!");}
}

由于URLClassLoader具有动态加载的特性,所以很适合拿来做热部署。

5.加载冲突

同一个类被不同加载器加载,加载结果视为两个不同类。

所以在加载之前需要判断是否被加载,如果未加载,则遵循双亲委托模型,从而避免加载冲突。

代码示例:

import java.io.File;
import java.net.URL;
import java.net.URLClassLoader;public class Main {public static void main(String[] args) throws Exception {// 创建两个不同的URLClassLoader,分别加载同一个类URL[] classpath = { new File("classes/").toURI().toURL() };URLClassLoader classLoader1 = new URLClassLoader(classpath);URLClassLoader classLoader2 = new URLClassLoader(classpath);// 使用classLoader1加载HelloWorld类Class helloWorldClass1 = classLoader1.loadClass("HelloWorld");// 使用classLoader2加载HelloWorld类Class helloWorldClass2 = classLoader2.loadClass("HelloWorld");// 输出两个类是否相同System.out.println(helloWorldClass1 == helloWorldClass2); // false}
}class HelloWorld {public void sayHello() {System.out.println("Hello, world!");}
}

相关内容

热门资讯

安卓无用系统文件清理,提速无忧 手机里的文件是不是越来越多,感觉就像一个乱糟糟的房间,东西堆得满满的,连个下脚的地方都没有?别急,今...
安卓系统如何修改头像,轻松修改... 亲爱的手机控们,你们有没有想过,为什么你的手机里那些可爱的头像总是那么吸引人呢?是不是也想给自个儿的...
安卓系统是中国的,发展历程、市... 你有没有想过,那个陪伴我们每天刷手机、玩游戏、看视频的安卓系统,竟然有着中国血统呢?没错,安卓系统,...
安卓系统文件重命名,安卓系统文... 手机里的文件重命名,这可是个让人头疼的小问题。有时候,你下载了一个文件,发现后缀名不对,想改一改,结...
使用安卓系统的比例,轻松驾驭各... 你知道吗?在咱们这个科技飞速发展的时代,手机已经成了我们生活中不可或缺的好伙伴。而说到手机,安卓系统...
安卓系统ins怎么下载,并且没... 安卓系统Ins怎么下载?揭秘Instagram安装全攻略Hey,亲爱的安卓手机用户们,你是否曾在某个...
照片怎么隐身安卓系统,揭秘安卓... 亲爱的手机控们,你是否有过这样的烦恼:一些私密的照片或视频,不想让他人随意翻看,但又不想删除,该怎么...
安卓系统 辣妈日记,安卓系统下... 亲爱的辣妈们,你们是不是也在寻找一款能记录宝宝成长点滴、记录自己心路历程的神奇软件呢?今天,就让我来...
安卓系统游戏账号在ios系统,... 亲爱的游戏迷们,你是不是也和我一样,手机换了个新伙伴,却苦于安卓系统上的游戏账号怎么在iOS系统上登...
安卓老机器系统吗,探索安卓老机... 家里的老电脑是不是已经闲置好久了?别急着扔掉,今天就来给你支个招,让你的老电脑焕发第二春,变身安卓小...
开源平板电脑安卓系统,引领智能... 你有没有想过,为什么你的平板电脑那么聪明,能陪你玩游戏、看电影,还能帮你处理工作?这都得归功于它背后...
安卓系统源码分析工具,基于安卓... 你有没有想过,安卓系统这个看似庞大而复杂的家伙,其实背后有着一套精心设计的源码体系?没错,今天咱们就...
海尔统帅电视安卓系统,畅享无限... 亲爱的电视迷们,今天我要和你聊聊一款超级实用的电视——海尔统帅电视,还有它那强大的安卓系统!想象坐在...
nova 7 pro安卓系统,... 你有没有注意到,最近你的手机是不是有点儿不一样了?没错,我要说的就是那个华为nova 7 Pro,这...
安卓系统埋雷排雷外挂,安卓系统... 你有没有发现,手机里的安卓系统有时候就像个“雷区”,一不小心就踩到了那些让人头疼的“地雷”?别急,今...
搜象文学安卓系统,探索文学新境... 亲爱的读者,你是否曾在手机上翻找那些充满想象力的文学作品,却因为系统的不兼容而感到沮丧?今天,我要给...
华为会替换安卓系统,全面迈向自... 你知道吗?最近手机圈里可是炸开了锅,因为华为要大动作了!没错,就是那个我们每天都离不开的安卓系统,华...
安卓系统查看功能设置,安卓系统... 亲爱的手机控们,你是否曾对着你的安卓手机,一脸懵圈,不知道怎么找到那些隐藏的宝藏功能呢?别急,今天我...
安卓系统化变苹果系统,无需换机 亲爱的手机控们,你是否曾幻想过,把你的安卓手机变成苹果手机,体验一下那传说中的流畅和优雅呢?别急,今...
小米原生安卓系统图片,功能与美... 你知道吗?在科技飞速发展的今天,手机操作系统已经成为衡量一款手机是否强大的重要标准。而小米,这个以性...