网站首页 > java教程 正文
类加载的基本机制和过程
运行Java程序,就是执行java这个命令,指定包含main方法的完整类名,以及一个classpath,即类路径。类路径可以有多个,对于直接的class文件,路径是class文件的根目录,对于jar包,路径是jar包的完整名称(包括路径和jar包名)。
Java运行时,会根据类的完全限定名寻找并加载类,寻找的方式基本就是在系统类和指定的类路径中寻找,如果是class文件的根目录,则直接查看是否有对应的子目录及文件;如果是jar文件,则首先在内存中解压文件,然后再查看是否有对应的类。
负责加载类的类就是类加载器,它的输入是完全限定的类名,输出是Class对象。类加载器不是只有一个,一般程序运行时,都会有三个(适用于Java 9之前,Java 9引入了模块化,基本概念是类似的,但有一些变化,限于篇幅,就不探讨了)。
1)启动类加载器
(Bootstrap ClassLoader):这个加载器是Java虚拟机实现的一部分,不是Java语言实现的,一般是C++实现的,它负责加载Java的基础类,主要是<JAVA_HOME>/lib/rt.jar,我们日常用的Java类库比如String、ArrayList等都位于该包内。
2)扩展类加载器
(Extension ClassLoader):这个加载器的实现类是sun.misc.Laun-cher$ExtClassLoader,负责加载Java的一些扩展类,一般是<JAVA_HOME>/lib/ext目录中的jar包。
3)应用程序类加载器
(Application ClassLoader):这个加载器的实现类是sun.misc.Launcher$AppClassLoader,它负责加载应用程序的类,包括自己写的和引入的第三方法类库,即所有在类路径中指定的类。
这三个类加载器有一定的关系,可以认为是父子关系,Application ClassLoader的父亲是Extension ClassLoader,Extension的父亲是Bootstrap ClassLoader。注意不是父子继承关系,而是父子委派关系,子ClassLoader有一个变量parent指向父ClassLoader,在子Class-Loader加载类时,一般会首先通过父ClassLoader加载,具体来说,在加载一个类时,基本过程是:
1)判断是否已经加载过了,加载过了,直接返回Class对象,一个类只会被一个Class-Loader加载一次。
2)如果没有被加载,先让父ClassLoader去加载,如果加载成功,返回得到的Class对象。
3)在父ClassLoader没有加载成功的前提下,自己尝试加载类。
这个过程一般被称为“双亲委派” 模型,即优先让父ClassLoader去加载。为什么要先让父ClassLoader去加载呢?这样,可以避免Java类库被覆盖的问题。比如,用户程序也定义了一个类java.lang.String,通过双亲委派,java.lang.String只会被Bootstrap ClassLoader加载,避免自定义的String覆盖Java类库的定义。
需要了解的是,“双亲委派”虽然是一般模型,但也有一些例外,比如:
1)自定义的加载顺序:
尽管不被建议,自定义的ClassLoader可以不遵从“双亲委派”这个约定,不过,即使不遵从,以java开头的类也不能被自定义类加载器加载,这是由Java的安全机制保证的,以避免混乱。
2)网状加载顺序:
在OSGI框架和Java 9模块化系统中,类加载器之间的关系是一个网,每个模块有一个类加载器,不同模块之间可能有依赖关系,在一个模块加载一个类时,可能是从自己模块加载,也可能是委派给其他模块的类加载器加载。
3)父加载器委派给子加载器加载:
典型的例子有JNDI服务(Java Naming and Directory Interface),它是Java企业级应用中的一项服务,具体我们就不介绍了。
一个程序运行时,会创建一个Application ClassLoader,在程序中用到ClassLoader的地方,如果没有指定,一般用的都是这个ClassLoader,所以,这个ClassLoader也被称为系统类加载器
(System ClassLoader)。
猜你喜欢
- 2025-02-04 SpringBoot开发 - 什么是热部署和热加载?devtool的原理是什么?
- 2025-02-04 JVM详解之:类的加载链接和初始化(类加载java)
- 2025-02-04 面试官:什么是java类加载当中的双亲委派?
- 2025-02-04 jvm超详细探索自定义类加载器(值得收藏)
- 2025-02-04 详解java反射的原理(java 的反射)
- 2025-02-04 JVM性能调优(1)——JVM内存模型和类加载运行机制
- 2025-02-04 Java反射:作用与原理解析(java反射到底有什么用)
- 2025-02-04 java类加载与初始化(java类的加载机制及加载过程)
- 2025-02-04 类加载的验证阶段你不知道的东西(类加载检查)
- 2025-02-04 类是如何加载的?(类的加载过程是什么?简单描述一下每个步骤)
你 发表评论:
欢迎- 最近发表
-
- Java常量定义防暴指南:从"杀马特"到"高富帅"的华丽转身
- Java接口设计原则与实践:优雅编程的艺术
- java 包管理、访问修饰符、static/final关键字
- Java工程师的代码规范与最佳实践:优雅代码的艺术
- 编写一个java程序(编写一个Java程序计算并输出1到n的阶乘)
- Mycat的搭建以及配置与启动(mycat部署)
- Weblogic 安装 -“不是有效的 JDK Java 主目录”解决办法
- SpringBoot打包部署解析:jar包的生成和结构
- 《Servlet》第05节:创建第一个Servlet程序(HelloSevlet)
- 你认为最简单的单例模式,东西还挺多
- 标签列表
-
- java反编译工具 (77)
- java反射 (57)
- java接口 (61)
- java随机数 (63)
- java7下载 (59)
- java数据结构 (61)
- java 三目运算符 (65)
- java对象转map (63)
- Java继承 (69)
- java字符串替换 (60)
- 快速排序java (59)
- java并发编程 (58)
- java api文档 (60)
- centos安装java (57)
- java调用webservice接口 (61)
- java深拷贝 (61)
- 工厂模式java (59)
- java代理模式 (59)
- java.lang (57)
- java连接mysql数据库 (67)
- java重载 (68)
- java 循环语句 (66)
- java反序列化 (58)
- java时间函数 (60)
- java是值传递还是引用传递 (62)
本文暂时没有评论,来添加一个吧(●'◡'●)