JVM总括四-类加载过程、双亲委派模型、对象实例化过程
JVM总括四-类加载过程、双亲委派模型、对象实例化过程
目录:JVM总括:目录
一、 类加载过程
类加载过程就是将.class文件转化为Class对象,类实例化的过程,(User user = new User(); 这个过程是对象实例化的过程);
一个.class文件只有一个Class对象(字节码对象),可以有无数个对象(例如:new User(););
1、Load:
将编译后的.class文件以二进制流的方式加载到JVM内存中,并转化为特定的数据结构,用到的就是classLoad二类加载器。这个过程中校验cafe babe魔法数、常量池、文件长度、是否有父类等。
其实加载阶段用一句话来描述就是:把代码数据加载到内存中。
2、Link:
分为验证、准备、解析三步。
验证:更为详细的验证,比如:final是否规范(二次赋值不规范)、static是否合理(静态方法必须引用静态变量)、类型是否正确。
准备:当完成字节码文件的校验之后,JVM 便会开始为类变量分配内存并初始化。这里需要注意两个关键点,即内存分配的对象以及初始化的类型。
内存分配的对象。Java 中的变量有「类变量」和「类成员变量」两种类型,「类变量」指的是被 static 修饰的变量,而其他所有类型的变量都属于「类成员变量」。
在准备阶段,JVM 只会为「类变量」分配内存,而不会为「类成员变量」分配内存。「类成员变量」的内存分配需要等到初始化阶段才开始。
比如下面的代码在准备阶段,只会为 age 属性分配内存,而不会为 website 属性分配内存。
public static int age= 3;
public String website = "www.beijingdesigner.com";
例如下面的代码在准备阶段之后,age的值将是 0,而不是 3。
public static int age= 3;
但如果一个变量是常量(被 static final 修饰)的话,那么在准备阶段,属性便会被赋予用户希望的值。例如下面的代码在准备阶段之后,number 的值将是 3,而不是 0。
public static final int number = 3;
这其中的道理我们想下就明白了。
两个语句的区别是一个有 final 关键字修饰,另外一个没有。被final修饰的常量,是不允许再被改变的。所以初始化就是3了。
解析:把类中的符号引用转化为直接引用,完成内存结构布局。(符号引用转化为直接引用:例如:test1() { test2(); },这里test1调用test2方法就是符号引用,但实际test2()通过一个指针指向test2方法的内存地址,这个指针负责调用,它就是直接引用)。
3、Init:
执行类构造器<clinit>,递归初始化父类的静态代码块,不执行构造函数。(先执静态变量,再执行行静态代码块)。
任何小写的class定义的类都有一个魔法属性:class,
- int j;
- Class<Integer> integerClass = int.class;
- Class<Integer> integerClass1 = Integer.class;
二、类加载的原则:双亲委派模型
类加载是怎么确定类文件的位置呢?
1、Bootstrap:最高级的类加载器,装置最核心的类,如:Object、System、String;
2、Extension ClassLoader:JDK9之前的类加载器,以后的为Platform ClassLoader,加载系统的扩展类;
3、Application ClassLoader:应用类加载器,主要加载用户自定义CLASSPATH路径下的类。
类加载器并非继承关系,只是以组合的方式服用父加载器功能,符合优先原则。
其中第二、第三层为java实现,用户可以自定义类加载器,第三层为C++实现,所有为null:
- ClassLoader classLoader = Son.class.getClassLoader();
- ClassLoader parent = classLoader.getParent();
- ClassLoader parent1 = parent.getParent();
- System.out.println(classLoader);
- System.out.println(parent);
- System.out.println(parent1);
- -----------------------------------
- sun.misc.Launcher$AppClassLoader@18b4aac2
- sun.misc.Launcher$ExtClassLoader@5acf9800
- null
获取Bootstrap加载的类库:
- URLClassPath bootstrapClassPath = Launcher.getBootstrapClassPath();
- URL[] urLs = bootstrapClassPath.getURLs();
- for (URL url : urLs){
- System.out.println(url.toExternalForm());
- }
---------------------------------------------------------- file:/D:/jdk-8u151-windows-x64/JDK/jre/lib/resources.jar
- file:/D:/jdk-8u151-windows-x64/JDK/jre/lib/rt.jar
- file:/D:/jdk-8u151-windows-x64/JDK/jre/lib/sunrsasign.jar
- file:/D:/jdk-8u151-windows-x64/JDK/jre/lib/jsse.jar
- file:/D:/jdk-8u151-windows-x64/JDK/jre/lib/jce.jar
- file:/D:/jdk-8u151-windows-x64/JDK/jre/lib/charsets.jar
- file:/D:/jdk-8u151-windows-x64/JDK/jre/lib/jfr.jar
- file:/D:/jdk-8u151-windows-x64/JDK/jre/classes
Bootstrap加载的路径可以追加,在JVM增加启动参数,不过不建议修改:
- Xbootclasspath/D:/test/src
自定义类加载器的情况:
1、隔离加载类:框架中吧类加载到不同的环境中
2、修改类加载方式:除了Bootstrap,其他类并非一定要加入。
3、扩展加载源:比如加载数据库;
4、防止源码泄露:可以对类进行加密,那加载的时候需要自定义加载器对类进行解密加载。
自定义ClassLoder,继承ClassLoader,重写findClass(),调用defineClass():
- public class UserClassLoader extends ClassLoader {
- private String classpath;
- public UserClassLoader(String classpath) {
- this.classpath = classpath;
- }
- @Override
- protected Class<?> findClass(String name) throws ClassNotFoundException {
- try {
- byte [] classDate=getDate(name);
- if(classDate==null){
- throw new FileNotFoundException();
- }
- else{
- //defineClass方法将字节码转化为类
- return defineClass(name,classDate,0,classDate.length);
- }
- } catch (IOException e) {
- e.printStackTrace();
- }
- return super.findClass(name);
- }
- //返回类的字节码
- private byte[] getDate(String className) throws IOException{
- InputStream in = null;
- ByteArrayOutputStream out = null;
- String path=classpath + File.separatorChar +className.replace('.',File.separatorChar)+".class";
- try {
- in=new FileInputStream(path);
- out=new ByteArrayOutputStream();
- byte[] buffer=new byte[2048];
- int len=0;
- while((len=in.read(buffer))!=-1){
- out.write(buffer,0,len);
- }
- return out.toByteArray();
- }
- catch (FileNotFoundException e) {
- e.printStackTrace();
- }
- finally{
- in.close();
- out.close();
- }
- return null;
- }
- }
UserClassLoader
调用,其中D:\\wp目录下为.class文件,里面有个Son.class
- //自定义类加载器的加载路径
- UserClassLoader myClassLoader=new UserClassLoader("D:\\wp");
- //包名+类名
- Class sonClass=myClassLoader.loadClass("com.Son");
- if(sonClass!=null){
- Object obj=sonClass.newInstance();
- System.out.println(sonClass.getClassLoader().toString());
- }
三、对象实例化
从字节码、执行步骤两个方面分析
1、字节码方面:
- Object object = new Object();,通过javap -verbose -p查看对象创建的字节码指令:
(1)new:如果找不到Class对象就进行类加载,然后分配内存(本类路径上所有的属性都分配),其中对象的引用也是个变量也占内存(4个字节),这个指令执行完毕会把对象的压入虚拟机栈顶。
(2)dup:在栈顶复杂引用,如果<init>有参数,把参数压入操作栈,两个引用,压入栈底的用来赋值或保存到局部变量表中,栈顶引用作为句柄调用相关方法。
(3)invokespecial:调用对象实例化方法,通过栈顶方法调用<init>方法(也就是调用构造方法)。
2、执行步骤
(1)确认类元信息是否存在:接到new指令时,在metaspace检查类元信息是否存在,没有就在双亲委派模式下进行类加载,生成Class对象。
(2)分配对象内存:首先计算对象占用空间大小(成员变量是引用变量就分配4个字节大小的变量空间),在堆中划分内存空间给新对象(分配空间需要进行同步操作,如:cas)。
(3)设定默认值:成员变量设置不同形式的0值;
(4)设置对象头:设置对象的哈希码、锁信息、对象所属的类元信息,设置取决于JVM。
(5)执行init方法:初始化成员变量,执行代码块,调用类的构造方法,把堆对象首地址赋值给引用变量。
四、思考:ClassLoader.loadClasshe和Class.forName区别
- //
- Class<Son> sonClass = Son.class;
- //
- Class<?> aClass = Maint.class.getClassLoader().loadClass("com.Son");
- //
- Class<?> bClass = Class.forName("com.Son");
代码2:
其实这种方法调运的是:ClassLoader.loadClass(name, false)方法
参数一:name,需要加载的类的名称
参数二:false,这个类加载以后是否需要去连接(不需要linking)
代码:3:
其实这种方法调运的是:Class.forName(className, true, ClassLoader.getCallerClassLoader())方法
参数一:className,需要加载的类的名称。
参数二:true,是否对class进行初始化(需要initialize)
参数三:classLoader,对应的类加载器
其中1、2都是将.class文件加载到JVM中,得到Class对象,但是还没有连接(Link),静态代码块不会执行;
代码3得到Class对象并且已经完成初始化<clinit>,静态代码块会执行。( Class.forName("com.mysql.jdbc.Driver");//通过这种方式将驱动注册到驱动管理器上)。
1、2、3都不会执行对象的构造函数。
五、思考:<clinit>和<init>区别
<clinit>是类(Class)初始化执行的方法,<init>是对象初始化执行的方法(构造函数);
JVM总括四-类加载过程、双亲委派模型、对象实例化过程的更多相关文章
- JVM探究之 —— 类加载器-双亲委派模型
虚拟机设计团队把类加载阶段中的“通过一个类的全限定名来获取描述此类的二进制字节流”这个动作放到Java虚拟机外部去实现,以便让应用程序自己决定如何去获取所需要的类.实现这个动作的代码模块称为“类加载器 ...
- JVM(16)之 双亲委派模型
开发十年,就只剩下这套架构体系了! >>> 在上一篇博文中,我们知道了如何获得二进制的字节流,并根据获得的字节流去装载一个类.同时也了解到类加载器的存在,每个加载器对应着不同的加 ...
- 深入理解JVM(③)虚拟机的类加载器(双亲委派模型)
前言 先解释一下什么是类加载器,通过一个类的全限定名来获取描述该类的二进制字节流,在虚拟机中实现这个动作的代码被称为"类加载器(Class Loader)". 类与类加载器 类加载 ...
- 类文件的结构、JVM 的类加载过程、类加载机制、类加载器、双亲委派模型
一.类文件的结构 我们都知道,各种不同平台的虚拟机,都支持 "字节码 Byte Code" 这种程序存储格式,这构成了 Java 平台无关性的基石.甚至现在平台无关性也开始演变出 ...
- JVM类加载过程与双亲委派模型
类加载过程 类加载过程为JVM将类描述数据从.class文件中加载到内存,并对数据进行解析和初始化,最终形成被JVM直接使用的Java类型.包含: 加载:获取该类的二进制字节流,将字节流代表的静态存储 ...
- JVM的类加载过程以及双亲委派模型详解
JVM的类加载过程以及双亲委派模型详解 这篇文章主要介绍了JVM的类加载过程以及双亲委派模型详解,类加载器就是根据指定全限定名称将 class 文件加载到 JVM 内存,然后再转化为 class 对象 ...
- JVM——类加载器的双亲委派模型
类加载器双亲委派模型,如下图所示: 双亲委派模型的工作过程 如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此,因此 ...
- jvm类加载器和双亲委派模型
类加载器按照层次,从顶层到底层,分为以下三种: (1)启动类加载器(Bootstrap ClassLoader) 这个类加载器负责将存放在JAVA_HOME/lib下的,或者被-Xbootcla ...
- JVM(四)JVM的双亲委派模型
1.两种不同的类加载器 从JAVA虚拟机的角度来讲,只存在两种不同的类加载器:一种是启动类加载器(Bootstrap ClassLoader),这个类加载器使用C++语言实现,是虚拟机自身的一部分:另 ...
随机推荐
- C# 数据库
连接: using System; using System.Collections.Generic; using System.Linq; using System.Text; using Syst ...
- multiprocessing还是threading?
今夜看了一篇分析python中多进程与多线程优劣的文章,文章通过几组性能测试强调了多进程的性能优势,同时也深入分析了为何python中多线程性能较差的原因,GIL就是解释器全局锁,该机制限制每个pyt ...
- docker入门 什么是docker? 为什么使用docker?
1.什么是docker? 轻量级操作系统虚拟化解决方案 2.为什么使用docker? 1.docker的启动是秒级的,比传统虚拟机快很多 2.资源利用率高,一台主机上可同时运行数千个docker容器 ...
- oracle入坑日记<一> 安装
学习日记系列(前辈/大神勿喷) 一.下载 下载地址:http://www.oracle.com/technetwork/cn/database/enterprise-edition/downloads ...
- python使用xlrd读取excel数据时,整数变小数的解决办法
python使用xlrd读取excel数据时,整数变小数: 解决方法: 1.有个比较简单的就是在数字和日期的单元格内容前加上一个英文的逗号即可.如果数据比较多,也可以批量加英文逗号的前缀(网上都有方法 ...
- Java -- XStreamAlias 处理节点中的属性和值
XStreamAlias 可以把objec和xml相互转换,但是有时候节点带有属性和值就需要特殊处理下: <?xml version="1.0" encoding=" ...
- mac 安装软件
一.安装spark 1.官网下载最新tar文件 2.解压 3.安装java开发环境 3.1.安装下载java 8 https://www.oracle.com/technetwork/java/ja ...
- RMI(远程方法调用)入门
这两篇可以入门 http://www.cnblogs.com/ninahan0419/archive/2009/06/25/javarmi.html http://www.cnblogs.com/wx ...
- Github使用笔记——创建远程库
系统:CentOS7 一.yum install git 二.配置 git config --global user.name "XXX" git config -global u ...
- Shell脚本21-40例
21.统计数字并求和 计算文档a.txt中每一行中出现的数字个数并且要计算一下整个文档中一共出现了几个数字.例如a.txt内容如下:12aa*lkjskdjalskdflkskdjflkjj 我们脚本 ...