后台(30)——类加载器ClassLoader

简介: 探索Android软键盘的疑难杂症 深入探讨Android异步精髓Handler 详解Android主流框架不可或缺的基石 站在源码的肩膀上全解Scroller工作机制Android多分...

探索Android软键盘的疑难杂症
深入探讨Android异步精髓Handler
详解Android主流框架不可或缺的基石
站在源码的肩膀上全解Scroller工作机制


Android多分辨率适配框架(1)— 核心基础
Android多分辨率适配框架(2)— 原理剖析
Android多分辨率适配框架(3)— 使用指南


自定义View系列教程00–推翻自己和过往,重学自定义View
自定义View系列教程01–常用工具介绍
自定义View系列教程02–onMeasure源码详尽分析
自定义View系列教程03–onLayout源码详尽分析
自定义View系列教程04–Draw源码分析及其实践
自定义View系列教程05–示例分析
自定义View系列教程06–详解View的Touch事件处理
自定义View系列教程07–详解ViewGroup分发Touch事件
自定义View系列教程08–滑动冲突的产生及其处理


版权声明


类加载器简介

在Java中类加载器ClassLoader负责加载所有的类,该类被加载至内存后会生成一个与之对应的java.lang.Class实例。当类被装载进JVM以后,当需要再次使用同一个类时就不用再次从内存外读取,而是直接使用JVM中的该类。嗯哼,那么怎么样才算是”同一个”类呢?或者说检验是否是”同一个”类的标准是什么呢?在Java中用类的全限定类名(包名+类名)唯一标识一个类;在JVM中用类的全限定类名(包名+类名)并结合加载它的类加载器共同唯一标识一个类。比如,tokyo.hot包下有个AVer类,该类被ClassLoader1加载进JVM;与此同时该类又被ClassLoader2加载进JVM;那么它们两者在JVM中对应的Class的唯一标识分别为(AVer,tokyo.hot,ClassLoader1)和(AVer,tokyo.hot,ClassLoader2)。这意味着同一个类被两个不同的类加载器加载至JVM时,这两个类彼此独立,互不影响是两个完全不同的Class

嗯哼,那么Java中是不是有个万能的类加载器ClassLoader可以将任何类加载进JVM呢?非也!!!JVM启动时会生成三个类加载器:

  • 根类加载器:Bootstrap ClassLoader
  • 扩展类加载器:Extension ClassLoader
  • 系统类加载器:System ClassLoader

现在,我们来依次学习这三个类加载器


Bootstrap ClassLoader

Bootstrap ClassLoader常被称为根(原始、引导)类加载器,负责加载Java的核心类。至于它具体加载了哪些Java核心类呢?我们来通过一个例子看看

/**
 * 本文作者:谷哥的小弟
 * 博客地址:http://blog.csdn.net/lfdfhl
 */
package cn.com;
import java.net.URL;
import sun.misc.URLClassPath;

public class TestBootstrap {
    public static void main(String[] args) {
        URLClassPath bootstrapClassPath = sun.misc.Launcher.getBootstrapClassPath();
        URL[] urls = bootstrapClassPath.getURLs();
        for (int i = 0; i < urls.length; i++) {
            System.out.println(urls[i].toExternalForm());
        }
    }
}

在编写这段代码时Eclipse可能会报错:Access restriction: The type ‘Launcher’ is not API。该问题可以这么解决:Project -> Properties -> libraries然后remove原来的JRE System Library,然后再Add Library重新添加一次即可。

打印结果如下:

file:/C:/Program Files/Java/jre1.8.0_60/lib/resources.jar
file:/C:/Program Files/Java/jre1.8.0_60/lib/rt.jar
file:/C:/Program Files/Java/jre1.8.0_60/lib/sunrsasign.jar
file:/C:/Program Files/Java/jre1.8.0_60/lib/jsse.jar
file:/C:/Program Files/Java/jre1.8.0_60/lib/jce.jar
file:/C:/Program Files/Java/jre1.8.0_60/lib/charsets.jar
file:/C:/Program Files/Java/jre1.8.0_60/lib/jfr.jar
file:/C:/Program Files/Java/jre1.8.0_60/classes

咦,看到这些,我们还是觉得很陌生。是么?来,我们一起来看看熟悉的

这里写图片描述

请打开rt.jar后再打开java.lang,嗯哼,是不是看到我们熟悉的东西了String.class、Thread.class、Integer.class、Object等等。这些Java核心类正是由Bootstrap ClassLoader加载的。


Extension ClassLoader

Extention ClassLoader被称为扩展类加载器,它负责加载JRE的扩展目录(%JAVA_HOME%/jre/lib/ext)中JAR包的类。所以,可将我们自己开发好的JAR包放入该路径下从而为java扩展核心类以外的功能。


System ClassLoader

System ClassLoader被称为系统(应用)类加载器,它负责在JVM启动时加载CLASSPATH所指定的JAR包和类路径。我们可通过ClassLoader的静态方法getSystemClassLoader()获取系统类加载器。一般情况下,我们编写的java类都是由这个类加载器加载。


类加载器的工作机制

刚才我们已经介绍了这三种类加载器,我们现在通过代码的形式来再次认识它们:

/**
 * 本文作者:谷哥的小弟
 * 博客地址:http://blog.csdn.net/lfdfhl
 */
package cn.com;

import java.io.IOException;
import java.net.URL;
import java.util.Enumeration;

public class TestClassLoader {
    public static void main(String[] args) throws IOException {
        // 获取系统类加载器
        ClassLoader classLoader = ClassLoader.getSystemClassLoader();
        System.out.println("系统类加载器SystemClassLoader="+classLoader);
        Enumeration<URL> urlEnumeration = classLoader.getResources("");
        while (urlEnumeration.hasMoreElements()) {
            URL nextElement = urlEnumeration.nextElement();
            System.out.println("系统类加载器SystemClassLoader的加载路径="+nextElement);
        }
        //获取扩展类加载器
        ClassLoader extLoader = classLoader.getParent();
        System.out.println("扩展类加载器ExtClassLoader=" + extLoader);
        String property = System.getProperty("java.ext.dirs");
        System.out.println("扩展类加载器ExtClassLoader的加载路径=" + property);
        //获取扩展类加载器的父类
        ClassLoader bootstrapClassLoader=extLoader.getParent();
        if(bootstrapClassLoader==null){
            System.out.println("扩展类加载器的父类=null");
        }
    }
}

输出如下:

系统类加载器SystemClassLoader=sun.misc.Launcher$AppClassLoader@4e0e2f2a
系统类加载器SystemClassLoader的加载路径=file:/D:/Workspace/TestClassLoader01/bin/
扩展类加载器ExtClassLoader=sun.misc.Launcher$ExtClassLoader@5c647e05
扩展类加载器ExtClassLoader的加载路径=C:\Program Files\Java\jre1.8.0_60\lib\ext;C:\WINDOWS\Sun\Java\lib\ext
扩展类加载器的父类=null

在这我们可以看出来:

  • 系统类加载器的加载路径是当前程序运行的路径

  • 扩展类加载器的加载路径是C:\Program Files\Java\jre1.8.0_60\lib\ext

  • 系统类加载器的父类是扩展类加载器

  • 扩展类加载器的getParent()返回值为null,并不是我们预计的根类加载器。这是因为根类加载器Bootstrap ClassLoader没有继承java.lang.ClassLoader抽象类。当然,实际上扩展类加载器的父类加载器就是根加载器,只不过根加载器不是Java实现的而是由JVM自身实现的。

在明白了这些之后,我们再来看类加载器中的父类委托机制。

当一个类加载器收到了类加载的请求时它不会自己去尝试加载这个类,而是把该请求委派给父加载器去完成。这样一层一层地往上传递,最终所有的加载请求都被传到最顶层的根类加载器中。只有当父加载器无法完成这个加载请求时,父加载器再将该请求交给子加载器去尝试加载。

这么做有什么好处呢?
当一个类加载器负责加载某个Class时,该Class所依赖和应该的其它Class也将由该类加载器载入。我们知道java.lang.Object是所有类的父类,因此无论哪个类加载都要加载这个类,于是关于加载Object的所有的请求都汇聚到了顶层的根类加载器中,因此Object类会由根类加载器来加载。所以在一次Object类后,就不用再次加载该类了。如果不采用该父类委托模式而由各个类加载器自行加载的话,系统中就会出现多个Object类,不仅效率低下而且混乱不堪也不能保证加载的对象是同一个。

相关文章
|
Linux
Win或Linux系统下用conda安装Open Babel
Win或Linux系统下用conda安装Open Babel
2110 0
Win或Linux系统下用conda安装Open Babel
|
存储 缓存 资源调度
想了解流计算,你必须得看一眼,实现Flink on Yarn的三种部署方式,并运行wordcount
想了解流计算,你必须得看一眼,实现Flink on Yarn的三种部署方式,并运行wordcount
1466 0
想了解流计算,你必须得看一眼,实现Flink on Yarn的三种部署方式,并运行wordcount
|
10月前
|
设计模式 前端开发 搜索推荐
前端必须掌握的设计模式——模板模式
模板模式(Template Pattern)是一种行为型设计模式,父类定义固定流程和步骤顺序,子类通过继承并重写特定方法实现具体步骤。适用于具有固定结构或流程的场景,如组装汽车、包装礼物等。举例来说,公司年会节目征集时,蜘蛛侠定义了歌曲的四个步骤:前奏、主歌、副歌、结尾。金刚狼和绿巨人根据此模板设计各自的表演内容。通过抽象类定义通用逻辑,子类实现个性化行为,从而减少重复代码。模板模式还支持钩子方法,允许跳过某些步骤,增加灵活性。
491 11
|
10月前
|
Kubernetes Cloud Native 微服务
探索云原生技术:容器化与微服务架构的融合之旅
本文将带领读者深入了解云原生技术的核心概念,特别是容器化和微服务架构如何相辅相成,共同构建现代软件系统。我们将通过实际代码示例,探讨如何在云平台上部署和管理微服务,以及如何使用容器编排工具来自动化这一过程。文章旨在为开发者和技术决策者提供实用的指导,帮助他们在云原生时代中更好地设计、部署和维护应用。
|
9月前
|
弹性计算 运维 安全
阿里云云服务诊断工具评测报告
阿里云云服务诊断工具评测报告
183 13
|
12月前
|
人工智能 运维 安全
AI来袭,IT人的工作还安全吗?来自汽车行业CIO的探讨
AI来袭,IT人的工作还安全吗?来自汽车行业CIO的探讨
AI来袭,IT人的工作还安全吗?来自汽车行业CIO的探讨
|
机器学习/深度学习 算法
m基于深度学习的32QAM调制解调系统相位检测和补偿算法matlab仿真
m基于深度学习的32QAM调制解调系统相位检测和补偿算法matlab仿真
262 1
|
存储
十六进制是什么
【8月更文挑战第9天】 十六进制是什么
2801 1
|
自然语言处理 程序员 Windows
[UE虚幻引擎] DTSpeechVoice 文字转语音播放 插件说明
这个插件用于在虚幻引擎(UE)中通过蓝图将文本转化为语音播放,利用Windows内置的语音引擎,支持Win10和Win11。确保电脑已安装语音系统,可能需要额外下载语言包以支持多语言播放。蓝图操作包括添加Speech Voice Component到Actor,使用Speak节点播放文本,Set Volume调整音量,Set Rate改变播放速度,Pause和Resume控制播放状态,Stop则停止播放且无法恢复。此外,Get Tokens和Set Token用于管理语音类型。更多详情可访问[80后程序员](https://dt.cq.cn/archives/1008?from=aliyun)
390 5
业技融合(BizDevOps)
该文档聚焦业技融合(BizDevOps),旨在构建业务与技术一体化的数字化组织,促进数字业务创新和持久发展。核心内容包括:1) 着重客户价值的协同;2) 全程数字化运营;3) 基于数据的透明度和效能衡量。实践中,涉及产品导向的团队结构、业务驱动的协作、研发资产管理、持续业务交付及数字业务创新。
247 2