ClassLoader&双亲委派&类初始化过程
1.class sycle
类加载的生命周期:加载(Loading)–>验证(Verification)–>准备(Preparation)–>解析(Resolution)–>初始化(Initialization)–>使用(Using)–>卸载(Unloading)。

关注点1: loading 将class 二进制文件加载到内存中
- 通过一个类的全限定名来获取定义此类的二进制字节流。
- 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构。
- 在java堆中生成一个代表这个类的java.lang.Class对象,做为方法区这些数据的访问入口。
加载阶段完成之后二进制字节流就按照虚拟机所需的格式存储在方区去中。
关注点2: verifaction 这一阶段的目的是为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求
- 文件格式验证:验证字节流是否符合Class文件格式的规范,并且能被当前版本的虚拟机处理
- 元数据验证:对字节码描述的信息进行语义分析,以确保其描述的信息符合java语言规范的要求。
- 字节码验证:这个阶段的主要工作是进行数据流和控制流的分析。任务是确保被验证类的方法在运行时不会做出危害虚拟机安全的行为。
- 符号引用验证:这一阶段发生在虚拟机将符号引用转换为直接引用的时候(解析阶段),主要是对类自身以外的信息进行匹配性的校验。目的是确保解析动作能够正常执行。
关注点3: preparation 对静态变量赋默认值,而不是初始值(目标指),准备阶段是正式为静态变量分配内存并设置初始值,这些内存都将在方法区中进行分配,这里的变量仅包括类变量(静态变量)不包括实例(成员)变量。
关注点4: resolution :解析是虚拟机将常量池的符号引用替换为直接引用的过程
- 符号引用:符号引用以一组符号来描述所引用的目标,符号可以是任意形式的字面量,只要使用时能无歧义地定位到目标即可。符号引用与虚拟机实现的内存布局无关,引用的目标并不一定已经加载到内存中。
- 直接引用:直接引用可以是直接指向目标的指针,相对偏移量或是一个能间接定位到目标的句柄。直接饮用是与内存布局相关的。
- 类或接口的解析
- 字段的解析
- 类方法解析
- 接口方法解析
关注点5: initializing 负责执行类中的静态初始化代码、构造器代码以及静态属性的初始化(目标值)initializing 负责执行类中的静态初始化代码、构造器代码以及静态属性的初始化(目标值)
- 遇到new、getstatic、putstatic、invokestatic这4个字节码指令时,如果类没有进行过初始化,出发初始化操作。 访问final 变量除外 ??
- 使用java.lang.reflect包的方法对类进行反射调用时。
- 当初始化一个类的时候,如果发现其父类还没有执行初始化则进行初始化。
- 虚拟机启动时用户需要指定一个需要执行的主类,虚拟机首先初始化这个主类。
- 动态语言支持java,lang.invoke.MethodHandle解析结果为REF_getstatic REF_invokestatic的方法句柄时,该类必须初始化。
注意:接口与类的初始化规则在第三点不同,接口不要气所有的父接口都进行初始化。
2 不同类加载器说明
引导类加载器(BootStrap) :
主要负责加载JVM自身需要的类,该加载器由C++实现,加载的是<JAVA_HOME>/lib 下的class文件,或者 -Xbootclasspath 参数指定的路径下的jar包,注意必须由虚拟机按照文件名识别加载jar包,如rt.jar,如果文件名不被虚拟机识别,即使把jar丢到lib目录下也是没有最用的(出于考虑,Bootstrap 启动类加载器只加载java、javax、sun开头的类),引导类加载器在hotspot 虚拟中使用C++语言实现,它是虚拟机的一部分。除了引导类加载器之外,其他类加载器都是由Java语言实现,并且全部集成自java.lang.ClassLoader,他们是独立于虚拟机外部的。
扩展类加载器(Extension) :
扩展类加载是指Sun公司实现的类,它是由Sun的ExtClassLoader实现的,是Lancher类的静态内部类。他负责加载<JAVA_HOME>/lib/ext目录下或有系统变量-Djava.ext.dir指定路径中的类库,开发者可以直接使用标准扩展类加载器
public class Launcher {
......
static class ExtClassLoader extends URLClassLoader {
private static volatile Launcher.ExtClassLoader instance;
public static Launcher.ExtClassLoader getExtClassLoader() throws IOException {
if (instance == null) {
Class var0 = Launcher.ExtClassLoader.class;
synchronized(Launcher.ExtClassLoader.class) {
if (instance == null) {
instance = createExtClassLoader();
}
}
}
return instance;
}
}
.....
}
系统类加载器(应用程序加载器AppClassLoader):
它是由Sun的AppClassLoader实现的,它负责加载系统路径 java -classpath 或者-D java.class.path指定路径下的类库,也就是我们经常使用到的classpath路径,开发者直接使用系统类的加载器,一般情况下该类加载器是程序组中默认的类加载器,通过ClassLoad.getSystemClassLoader()方法可以获取到该类的加载器。
自定义类加载器(Custom ClassLoader ):
在程序运行期间, 通过java.lang.ClassLoader的子类动态加载class文件, 体现java动态实时类装入特性
3.ClassLoader加载类过程(双亲委派)
JVM在加载类时默认采用的双亲委派机制。通俗讲,就是某个特定的类加载器在接到类加载器的请求时,受限将加载任务委传给父类加载器, 依次递归,如果父类加载器可以完成类的加载任务,就返回成功;只有父类加载器无法完成此加载器任务时,才去自己加载

4.ClassLoader加载类过程(双亲委派流程图)

5.为什么需要双亲委派机制?
为了系统类的安全,类似“java.lang.Object”这种核心类,JVM需要保证他们生成的对象都会被认定为同一类型 ,如果用户编写了一个lava.lang.Object的同名类并放在ClassPath中,多个类加载器都去加载这个类到内存中,系统中会出现多个不同的Object类,那么类之间的比较傲结果以及唯一性将无法保证,并且如果不使用这种双亲委派模型将会给虚拟机的安全带来安全隐患。所以要让类对象进行比较有意义,前提是他们要被同一个类加载器加载。即“通过代理模式,对于java核心类库的类的加载工作由引导类加载器统一完成,保证了Java应用所使用的都是同一个版本的Java 核心库的类,是相互兼容的”
好处是防止内存中出现多份相同的字节码。
6.能不能自己写个类叫java.lang.System?答案:通常不可以,但可以采取另类方法达到这个需求。
解释:为了不让我们写System类,类加载采用委托机制,这样可以保证爸爸们优先,爸爸们能找到的类,儿子就没有机会加载。而System类是Bootstrap加载器加载的,就算自己重写,也总是使用Java系统提供的System,自己写的System类根本没有机会得到加载。
但是,我们可以自己定义一个类加载器来达到这个目的,为了避免双亲委托机制,这个类加载器也必须是特殊的。由于系统自带的三个类加载器都加载特定目录下的类,如果我们自己的类加载器加载一个特殊的目录,那么系统的加载器就无法加载,也就是最终还是由我们自己的加载器加载。
7.如何自定义类加载器
- 继承 ClassLoader
- overwrite findClass()
8.如何打破双亲委派?
- 集成ClassLoader
- 重写loadClass 方法
ClassLoader&双亲委派&类初始化过程的更多相关文章
- java安全沙箱(一)之ClassLoader双亲委派机制
java是一种类型安全的语言,它有四类称为安全沙箱机制的安全机制来保证语言的安全性,这四类安全沙箱分别是: 类加载体系 .class文件检验器 内置于Java虚拟机(及语言)的安全特性 安全管理器及J ...
- SpringBoot启动流程分析(一):SpringApplication类初始化过程
SpringBoot系列文章简介 SpringBoot源码阅读辅助篇: Spring IoC容器与应用上下文的设计与实现 SpringBoot启动流程源码分析: SpringBoot启动流程分析(一) ...
- 一道题反映Java的类初始化过程
Java的类初始化过程: 1. 父类的static成员变量,static语句块. 2. 子类的static成员变量,static语句块. 3. 父类的普通成员变量,构造函数. 4. 子类的普通成员变量 ...
- 【Java_基础】java类加载过程与双亲委派机制
1.类的加载.连接和初始化 当程序使用某个类时,如果该类还未被加载到内存中,则系统会通过加载.连接.初始化三个步骤来对类进行初始化.如果没有意外,jvm将会连续完成这三个步骤,有时也把这三个步骤统称为 ...
- 解析Java类和对象的初始化过程
类的初始化和对象初始化是 JVM 管理的类型生命周期中非常重要的两个环节,Google 了一遍网络,有关类装载机制的文章倒是不少,然而类初始化和对象初始化的文章并不多,特别是从字节码和 JVM 层次来 ...
- [转载]解析 Java 类和对象的初始化过程
原文地址:http://www.ibm.com/developerworks/cn/java/j-lo-clobj-init/index.html 由一个单态模式引出的问题谈起 类的初始化和对象初始化 ...
- Java类初始化和实例初始化过程
1.类初始化过程 一个类要创建实例需要先加载并初始化该类 main方法所在的类需要先加载和初始化 一个子类要初始化需要先初始化父类 一个类初始化就是执行<client>()方法(编译器生成 ...
- Java 类初始化和实例初始化过程
1.类初始化过程 2.实例初始化过程 3.方法的重写
- jvm类加载器以及双亲委派
首先来了解几个概念: 类加载: 概念:虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验--转换解析--初始化,最终形成能被java虚拟机直接使用的java类型,就是jvm的类加载机制. ...
随机推荐
- application.properties和application.yml
1.application.properties 文件和 application.yml 文件优先级: 当同时存在时,那application.yml 只是个摆设. 2.yml文件的好处,天然的树状结 ...
- 基于Springboot注解的策略模式
释义 策略模式和多态很相似 可以理解为定义了一个统一的接口,有许多不同的实现类,可以自由选择不同的实时类去执行. 实现 上代码: 定义一个统一的接口: [JavaScript] 纯文本查看 复制代码 ...
- VRRP笔记二:配置keepalived为实现haproxy高可用的配置文件示例
! Configuration File for keepalived global_defs { notification_email { linuxedu@foxmail.com mageedu@ ...
- 默认的Settings.xml文件(无修改过)-Maven
Tip: 当什么都不作修改时,默认是从Maven中央仓库进行下载,https://repo.maven.apache.org/maven2. 打开maven源码下的lib文件夹,找到maven-mod ...
- Image图片
# View more python tutorials on my Youtube and Youku channel!!! # Youtube video tutorial: https://ww ...
- 醉酒驾驶VS睡眠不足,哪个更危险
在全球范围内,睡眠不足驾驶每年导致成千上万起交通事故,醉酒驾驶就更不用说了.毫无疑问,两种情况下都不应该驾车,都有危险.睡眠不足和醉酒驾驶相对来说哪个更危险呢? 如果要问哪一个"好" ...
- makefile(3)函数
前言 学习make和makefile的主要目的是分析大型项目的源代码的关系,上一节我们讲述了makefile 中的变量,本节主要学习一下 makefile 中的函数,首先函数肯定可以分为几部分: 内置 ...
- spring入门-整合junit和web
整合Junit 导入jar包 基本 :4+1 测试:spring-test-5.1.3.RELEASE.jar 让Junit通知spring加载配置文件 让spring容器自动进行注入 1234567 ...
- Scrum模拟微信看一看“疫情专区”的敏捷开发过程
无论作为产品用户还是管理咨询顾问,都非常非常喜欢微信.自认感情比较克制属于“高冷”挂,但从很多方面都太佩服太崇拜张小龙了(新书里微信也会是最喜欢的案例之一,真的不只是一个产品而已,很多方面都太牛了). ...
- iMX287A多种方法实现流水灯效果
目录 1.流水灯在电子电路中的地位 2.硬件电路分析 3.先点个灯吧 4.shell脚本实现流水灯 5.ANSI C文件操作实现流水灯 6.Linux 系统调用实现流水灯 @ 1.流水灯在电子电路中的 ...