JVM 心得分享(加载 链接 初始化)

基本概念:类加载的过程大致分为三个阶段

1、加载阶段:本阶段主要把class的二进制代码加载进入JVM,并且进行常量池(类名,方法名,字段名),方法区(二进制字节码),栈(本地方法栈结构),堆(java.lang.class对象)的设置。

有三个加载类:Bootstrap ClassLoader,加载jre/lib/下的类;

Extension ClassLoader:加载jre/lib/ext下的类;

ApplicationClassLoader:加载classpath下的类(应用程序自己开发的类,如 工程目录/bin/下的.class文件)

还有一个扩展的加载类,满足应用程序的特殊需求。类的加载时,父亲loader优先执行load动作,父亲load不了时,子类运作。

2、链接阶段:又分为三个小阶段 校验,准备,解析。

校验:实施字节码文件的格式,语法等的校验。

准备:对静态变量申请存储空间,并设置默认的初始值。如:private static int a =2;那么在准备阶段a被设置为0;

解析:把方法区中的符号指针替换为直接引用。

3、初始化阶段:对静态变量进行初始化,执行静态块,创建类的实例。上述的a变量在初始化阶段会被设置为2。

第一步:验证静态变量和静态块的加载+链接(校验,准备,解析)+初始化过程中值的变化。

 package com.chong.studyparalell.clazz.loader;

public class ClassLoaderDemo {
public static void main(String []args){
Test test2 = new Test();

System.out.println("Test2实例化结束"+test2.toString());

}
}
package com.chong.studyparalell.clazz.loader;

public class Test{

private static Test test1 = new Test();
private static int a = 2;
private static int b = 2;

static {
System.out.println("【Test类静态块】a=" + a);
}

public Test(){

System.out.println("【Test类构造方法】a=" + a);
System.out.println("【Test类构造方法】b=" + b);
System.out.println("【Test类实例】" + this.toString());
}

public static Test newInstance(){
return test1;
}
}

log输出如下:

1 【Test类构造方法】a=0
2 【Test类构造方法】b=0
3 【Test类实例】com.chong.studyparalell.clazz.loader.Test@16c1857
4 【Test类静态块】a=2
5 【Test类构造方法】a=2
6 【Test类构造方法】b=2
7 【Test类实例】com.chong.studyparalell.clazz.loader.Test@1b1fd9c
8 Test2实例化结束com.chong.studyparalell.clazz.loader.Test@1b1fd9c

首先Test类在链接阶段(准备阶段),a,b分别被设置默认值0。

当new Test()执行后,

1)首先初始化Test类的三个静态变量 test1,a,b。

初始化test1时,第一次调用构造方法,此时a,b为0。对应日志1,2行。

实例化test1,日志第3行。

test1初始化完成后,继续初始化a,b,设为2。

接着初始化静态块 ,对应日志第4行。

2)执行Test类的构造方法

因为a,b已经被初始化为2,所以执行类的构造方法时,会输出a,b 为2。日志第5,6行。

实例化后输出地址信息,日志第7行。

3)最终main方法里打出实例工作完成,日志第8行。

第二步,加入父类后,进行确认。

package com.chong.studyparalell.clazz.loader;

public class TestBase {

private static int base_a = 2;
private static int base_b = 2;

static {
System.out.println("【父类静态块】 base_a="+base_a);
}

public TestBase(){
System.out.println("【父类 构造方法】base_a=" + base_a);
System.out.println("【父类 构造方法】base_b=" + base_b);
System.out.println("【父类 实例】"+ this.toString());
}

}
package com.chong.studyparalell.clazz.loader;

public class Test extends TestBase{
内容同第一步;
}

log输出如下:

【父类静态块】 base_a=2

【父类 构造方法】base_a=2

【父类 构造方法】base_b=2

【父类 实例】com.chong.studyparalell.clazz.loader.Test@19ab8d

【Test类构造方法】a=0

【Test类构造方法】b=0

【Test类实例】com.chong.studyparalell.clazz.loader.Test@19ab8d

【Test类静态块】a=2

【父类 构造方法】base_a=2

【父类 构造方法】base_b=2

【父类 实例】com.chong.studyparalell.clazz.loader.Test@14dcfad

【Test类构造方法】a=2

【Test类构造方法】b=2

【Test类实例】com.chong.studyparalell.clazz.loader.Test@14dcfad

Test2实例化结束com.chong.studyparalell.clazz.loader.Test@14dcfad

可以发现父类的静态变量,静态块,构造方法首先被初始化。然后子类的静态变量,静态块和构造方法被初始化。

第三步:写一个自定义的类加载器

 package com.chong.studyparalell.clazz.loader;

public class MyClassLoaderPilot {

public static void main(String[] args) {

try {
MyClassLoader classLoader = new MyClassLoader();
String filename = "com.chong.studyparalell.demon.DemonThreadDemo";

Object clazz = classLoader.loadCustomizeClass(filename);
System.out.println(clazz);

} catch (Exception e) {
e.printStackTrace();
}
}

}
package com.chong.studyparalell.clazz.loader;

import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.io.IOException;

public class MyClassLoader extends ClassLoader {

private String demoPath = "D:\\work\\temp\\";

public Class<?> loadCustomizeClass(String filename) throws ClassNotFoundException {

FileInputStream fis = null;
ByteArrayOutputStream baos = new ByteArrayOutputStream();
try {
// 1.获取class文件的字节码(二进制数据)
String[] fileNames = filename.split("\\.");
fis = new FileInputStream(demoPath + fileNames[fileNames.length-1] +".class");
byte[] bytes = new byte[1024];
int len = 0;
while ((len = fis.read(bytes)) != -1) {
baos.write(bytes, 0, len);
}

} catch (Exception e) {
throw new ClassNotFoundException();
} finally {
try {
fis.close();
} catch (IOException e) {
throw new ClassNotFoundException();
}
}

byte[] paramArrayOfByte = baos.toByteArray();
// 2。把二进制文件定义为class对象返回
return defineClass(filename, paramArrayOfByte, 0, paramArrayOfByte.length);
}
}

日志输出如下:

class com.chong.studyparalell.demon.DemonThreadDemo

实际的跟着代码走一遍,看看控制台的输出,用自己的思路虚拟着跟一跟,对于类的加载过程能够认识的更加清晰一些。

以上这篇JVM 心得分享(加载 链接 初始化)就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • JVM加载一个类的过程

    类的加载过程 Java源代码被编译成class字节码,JVM把描述类数据的字节码.Class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的java类型,这就是虚拟机的类加载机制. 类从被加载到虚拟机内存中开始,到卸载出内存为止,它的生命周期包括了:加载(Loading).验证(Verification).准备(Preparation).解析(Resolution).初始化(Initialization).使用(Using).卸载(Unloading)七个阶段,

  • JVM类加载机制详解

    一.先看看编写出的代码的执行过程: 二.研究类加载机制的意义 从上图可以看出,类加载是Java程序运行的第一步,研究类的加载有助于了解JVM执行过程,并指导开发者采取更有效的措施配合程序执行. 研究类加载机制的第二个目的是让程序能动态的控制类加载,比如热部署等,提高程序的灵活性和适应性. 三.类加载的一般过程 原理:双亲委托模式 1.寻找jre目录,寻找jvm.dll,并初始化JVM: 2.产生一个Bootstrap Loader(启动类加载器): 3.Bootstrap Loader自动加载E

  • 深入理解Java 类加载全过程

    Java类加载全过程 一个java文件从被加载到被卸载这个生命过程,总共要经历4个阶段: 加载->链接(验证+准备+解析)->初始化(使用前的准备)->使用->卸载 其中加载(除了自定义加载)+链接的过程是完全由jvm负责的,什么时候要对类进行初始化工作(加载+链接在此之前已经完成了),jvm有严格的规定(四种情况): 1.遇到new,getstatic,putstatic,invokestatic这4条字节码指令时,加入类还没进行初始化,则马上对其进行初始化工作.其实就是3种情况

  • JVM 心得分享(加载 链接 初始化)

    基本概念:类加载的过程大致分为三个阶段 1.加载阶段:本阶段主要把class的二进制代码加载进入JVM,并且进行常量池(类名,方法名,字段名),方法区(二进制字节码),栈(本地方法栈结构),堆(java.lang.class对象)的设置. 有三个加载类:Bootstrap ClassLoader,加载jre/lib/下的类: Extension ClassLoader:加载jre/lib/ext下的类: ApplicationClassLoader:加载classpath下的类(应用程序自己开发

  • 浅谈JVM之类的加载链接和初始化

    加载 JVM可以分为三大部分,五大空间和三大引擎,要讲起来也不是特别复杂,先看下面的总体的JVM架构图. 从上面的图中,我们可以看到JVM中有三大部分,分别是类加载系统,运行时数据区域和Execution Engine. 加载就是根据特定名称查找类或者接口的二进制表示,并根据此二进制表示来创建类和接口的过程. 运行时常量池 我们知道JVM中有一个方法区的区域,在JDK8中,方法区的实现叫做元空间.这个元空间是存放在本地内存中的. 方法区中存放着每个class对应的运行时常量池. 当类或者接口创建

  • 浅谈Java类的加载,链接及初始化

    一 类生命周期 Loading Linking(Verification.Preparation.Resolution) Initializing 二 类加载器 1 图解 2 代码 package jvm; public class T002_ClassLoadLevel { public static void main(String[] args) { System.out.println(String.class.getClassLoader()); System.out.println(

  • JVM入门之类加载与字节码技术(类加载与类的加载器)

    1. 类加载阶段 1.1 加载阶段 将类的字节码载入方法区中,内部采用 C++ 的 instanceKlass 描述 java 类,它的重要 field 有: _java_mirror 即 java 的类镜像,例如对 String 来说,就是 String.class,作用是把 klass 暴 露给 java 使用 _super 即父类 _fields 即成员变量 _methods 即方法 _constants 即常量池 _class_loader 即类加载器 _vtable 虚方法表 _ita

  • JavaWeb中web.xml初始化加载顺序详解

    需求说明 做项目时,为了省事,起初把初始化的配置都放在每个类中 static加载,初始化配置一多,就想把它给整理一下,这里使用servlet中的init方法初始化. web.xml说明 首先了解下web.xml中元素的加载顺序: 启动web项目后,web容器首先回去找web.xml文件,读取这个文件 容器会创建一个 ServletContext ( servlet 上下文),整个 web 项目的所有部分都将共享这个上下文 容器将 转换为键值对,并交给 servletContext 容器创建 中的

  • Android App中实现图片异步加载的实例分享

    一.概述 一般大量图片的加载,比如GridView实现手机的相册功能,一般会用到LruCache,线程池,任务队列等:那么异步消息处理可以用哪呢? 1.用于UI线程当Bitmap加载完成后更新ImageView 2.在图片加载类初始化时,我们会在一个子线程中维护一个Loop实例,当然子线程中也就有了MessageQueue,Looper会一直在那loop停着等待消息的到达,当有消息到达时,从任务队列按照队列调度的方式(FIFO,LIFO等),取出一个任务放入线程池中进行处理. 简易的一个流程:当

  • 概述java虚拟机中类的加载器及类加载过程

    1. 类加载子系统 1.1 概述 类加载子系统负责从文件系统或者网络中加载Class文件,Class文件在文件开头有特定的文件标识 ClassLoader只负责class文件的加载,至于它是否可以运行,则由Execution Engine 决定 加载的类信息存放于一块成为 :方法区的内存空间,除了类的信息外,方法区中还会存放运行时常量池信息,可能还包括字符串字面量和数字常量(这部分常量信息是Class文件中常量池部分的内存映射) 字节码中的常量池加载到 方法区 -----> 运行时常量池信息 1

  • classloader类加载器_基于java类的加载方式详解

    基础概念 Classloader 类加载器,用来加载 Java 类到 Java 虚拟机中.与普通程序不同的是.Java程序(class文件)并不是本地的可执行程序.当运行Java程序时,首先运行JVM(Java虚拟机),然后再把Java class加载到JVM里头运行,负责加载Java class的这部分就叫做Class Loader. JVM本身包含了一个ClassLoader称为Bootstrap ClassLoader,和JVM一样,BootstrapClassLoader是用本地代码实现

  • tensorflow 加载部分变量的实例讲解

    tensorflow模型保存为saver = tf.train.Saver()函数,saver.save()保存模型,代码如下: import tensorflow as tf v1= tf.Variable(tf.random_normal([784, 200], stddev=0.35), name="v1") v2= tf.Variable(tf.zeros([200]), name="v2") saver = tf.train.Saver() with tf

  • 通过实例解析Java class文件编译加载过程

    一.Java从编码到执行 首先我们来看一下Java是如何从编码到执行的呢? 我们有一个x.java文件通过执行javac命令可以变成x.class文件,当我们调用Java命令的时候class文件会被装载到内存中,这个过程叫做classloader.一般情况下我们自己写代码的时候会用到Java的类库,所以在加载的时候也会把Java类库相关的类也加载到内存中.装载完成之后会调用字节码解释器和JIT即时编译器来进行解释和编译,编译完之后由执行引擎开始执行,执行引擎下面对应的就是操作系统硬件了.下图是大

随机推荐