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的类加载机制. ...
随机推荐
- Offer垂青于有准备的人——微软亚洲研究院实习生们的就业分享
编者按:一年一度的"求职大战"又拉开了序幕,如何在求职中掌握主动,更好地展现自己,最后抓住Offer?且听微软亚洲研究院三位实习生慢慢道来,Offer总会垂青于有准备的人. 廖振, ...
- H5页面如何引入vConsole
vConsole github地址vConsole 是腾讯开源的项目,这就简单的介绍一下使用 使用npm引入vconsole.min.js下载 vConsole 的最新版本.(不要直接下载 dev 分 ...
- 吴裕雄--天生自然HTML学习笔记:HTML 标题
在 HTML 文档中,标题很重要. HTML 标题 标题(Heading)是通过 <h1> - <h6> 标签进行定义的. <h1> 定义最大的标题. <h6 ...
- php结合Redis实现高并发下的秒杀抢购功能
实现思路 准备两个队列A和B,假设A队列的名称为stock,用于存放商品总库存信息,B队列的名称为users,用于存放抢购成功后的用户信息.每当有用户进行抢购操作时,先从A队列弹出一个元素,如果该元素 ...
- JS计算日期加天数后的日期(起始日期+有效天数=截至日期)
/** * 优惠券有效期 * startDate:起始日期 * valueTime:有效天数 */ function transferCouponValueTime(startDate,valueTi ...
- Ruby爬虫header发送cookie,nokogiri解析html数据
之前用php写过一个爬虫,同样是获取局域网的网站数据,这次我使用相同的网络环境,更低的电脑配置,使用ruby来再次爬虫,惊人的发现ruby使用自带的类库net/http爬取速度要远远超过php的cur ...
- Python 字符编码判断
题记 在获取中文字符的时候,如果出现乱码的情况,我们需要了解当前的字符串的编码形式.使用下面两种方法可以判断字符串的编码形式. 法一: isinstance(s, str) 用来判断是否为一般字符串 ...
- SQL命令汇总
order by rocketmq_id; 查找主从在同一IP的集群和节点2. select rocketmq_id,ip,port,type,count(*) as num from t_rock ...
- Angular4——7.表单处理
在Angular中存在两种表单处理方式: 模版驱动式表单 表单的数据模型是通过组件模版中的相关指令来定义的.由于使用这种方式定义表单的数据模型时,我们会受限于HTML的语法,所以,模版驱动方式只适用于 ...
- Ubuntu19.10安装OMNeT++ (omnetpp-5.6)中遇到的问题
在官网上下载对应版本的安装包,里面有说明性的文档,先在第五章ubuntu那里配置好前期的环境,再到linux那一章,看进行安装,本文即从这里开始记录. 安装包中的文档目录为:omnetpp-5.6/d ...