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&双亲委派&类初始化过程的更多相关文章

  1. java安全沙箱(一)之ClassLoader双亲委派机制

    java是一种类型安全的语言,它有四类称为安全沙箱机制的安全机制来保证语言的安全性,这四类安全沙箱分别是: 类加载体系 .class文件检验器 内置于Java虚拟机(及语言)的安全特性 安全管理器及J ...

  2. SpringBoot启动流程分析(一):SpringApplication类初始化过程

    SpringBoot系列文章简介 SpringBoot源码阅读辅助篇: Spring IoC容器与应用上下文的设计与实现 SpringBoot启动流程源码分析: SpringBoot启动流程分析(一) ...

  3. 一道题反映Java的类初始化过程

    Java的类初始化过程: 1. 父类的static成员变量,static语句块. 2. 子类的static成员变量,static语句块. 3. 父类的普通成员变量,构造函数. 4. 子类的普通成员变量 ...

  4. 【Java_基础】java类加载过程与双亲委派机制

    1.类的加载.连接和初始化 当程序使用某个类时,如果该类还未被加载到内存中,则系统会通过加载.连接.初始化三个步骤来对类进行初始化.如果没有意外,jvm将会连续完成这三个步骤,有时也把这三个步骤统称为 ...

  5. 解析Java类和对象的初始化过程

    类的初始化和对象初始化是 JVM 管理的类型生命周期中非常重要的两个环节,Google 了一遍网络,有关类装载机制的文章倒是不少,然而类初始化和对象初始化的文章并不多,特别是从字节码和 JVM 层次来 ...

  6. [转载]解析 Java 类和对象的初始化过程

    原文地址:http://www.ibm.com/developerworks/cn/java/j-lo-clobj-init/index.html 由一个单态模式引出的问题谈起 类的初始化和对象初始化 ...

  7. Java类初始化和实例初始化过程

    1.类初始化过程 一个类要创建实例需要先加载并初始化该类 main方法所在的类需要先加载和初始化 一个子类要初始化需要先初始化父类 一个类初始化就是执行<client>()方法(编译器生成 ...

  8. Java 类初始化和实例初始化过程

    1.类初始化过程 2.实例初始化过程 3.方法的重写

  9. jvm类加载器以及双亲委派

    首先来了解几个概念: 类加载: 概念:虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验--转换解析--初始化,最终形成能被java虚拟机直接使用的java类型,就是jvm的类加载机制. ...

随机推荐

  1. bug宝典linux篇 LC_CTYPE: cannot change locale (en_US.UTF-8): No such file or directory(转)

    升级glibc库后,使用xShell登录linux,提示: Connecting to ... Connection established. To escape to local shell, pr ...

  2. 吴裕雄--天生自然python学习笔记:Python3 JSON 数据解析

    JSON (JavaScript Object Notation) 是一种轻量级的数据交换格式.它基于ECMAScript的一个子集. Python3 中可以使用 json 模块来对 JSON 数据进 ...

  3. markdown常见用法

      命令  生成目录  [TOC]  插入标题  # 一级标题## 二级标题### 三级标题#### 四级标题##### 五级标题###### 六季标题  插入shell 开头:```shell 结尾 ...

  4. keep pace with |sixes and sevens.|Three dozen of |setting out|in despite of|appetite for|brought up|.turn to|leave behind|As can be seen|every

    Heavy but not excessive: network capacity seems to have done little more than keep pace with economi ...

  5. vue-cli 项目结构介绍

    感谢:https://www.jianshu.com/p/7006a663fb9f 总体框架 一个vue-cli的项目结构如下,其中src文件夹是需要掌握的,所以本文也重点讲解其中的文件,至于其他相关 ...

  6. 写了个通作的分页存储过程,top,加入了排序

    USE [WebDB_TradeOrder]GO/****** Object:  StoredProcedure [dbo].[Boss_Proc_PagingWithOrder]    Script ...

  7. centos7 开机/etc/rc.local 不执行的问题(转载)

    最近发现centos7 的/etc/rc.local不会开机执行,于是认真看了下/etc/rc.local文件内容的就发现了问题的原因了 #!/bin/bash # THIS FILE IS ADDE ...

  8. Pytorch collate_fn用法

    By default, Dataloader use collate_fn method to pack a series of images and target as tensors (first ...

  9. 关于KMP算法的重大发现

    之前写KMP模板的时候,nx[i]代表最大的一个x,使s[1,x-1]是s[1,i-1]的后缀.(方法1) 然而网上还有另一种方法求nx数组,nx[i]表示最大的一个x,使s[1,x]是s[1,i]的 ...

  10. python 堡垒机讲解及实例

    paramiko模块,该模块基于SSH用于连接远程服务器并执行相关操作. SSHClient:用于连接远程服务器并执行基本命令 #coding:utf-8 import paramiko ssh=pa ...