类加载概述

  1. 在java代码中,类型加载、连接初始化过程都是在程序运行期间完成的

类型:class、interface(object本身)、类型可在运行期间生成,如动态代理。一种runting概念

加载:最常见的方式是将已经存在的类的字节码文件(.class文件)从磁盘上加载到内存中;

连接:将类与类之间的关系确立下来,对于字节码的一些相关处理、校验在连接阶段完成

初始化:对于一些静态的变量进行复制

过程不是一定按照上述顺序,按照规范即可

  1. 提供了更大的灵活性,增加了更多的可能性

本身是静态类型的语言,但是很多特点又使得java具有动态的特点

类加载器深入剖析

  • Java虚拟机与程序生命周期: 在下面的情况下,Java虚拟机将结束生命周期

执行了System.exit()方法

程序正常结束

程序在执行过程中遇到了异常或者错误而异常终止

由于操作系统出现错误而导致Java虚拟机进程终止

类的加载、连接、初始化进一步介绍

  • 加载:查找并加载类的二进制数据

class文件(不一定是文件 也许来源网络 数据库等) 加载进内存

  • 连接
    • 验证:确保被加载的类的正确性
    • 准备:为类的静态变量分配内存,并将其初始化为默认值

此时类的实例对象还没创建

  1. //假设代码中有
  2. class Test{
  3. public static int a = 1;
  4. }
  5. //在准备这个阶段,虚拟机在准备阶段不会将1赋值给a,它会为a分配内存并且把0赋值给a
    • 解析:把类中的符号引用转换为直接引用

符号引用:间接的引用方式

直接引用:直接将引用的对象指向内存中的位置

  • 初始化:为类的静态变量赋予正确的初始化值
  1. //假设代码中有
  2. class Test{
  3. public static int a = 1;
  4. }
  5. //在准备阶段,虚拟机在准备阶段不会将1赋值给a,它会为a分配内存并且把0赋值给a
  6. //在初始化阶段,将a=0替换为a=1

类的使用与卸载

  • 类的使用

创建对象、使用方法

  • 类的卸载

从内存中销毁,osgi


类加载

主动使用与被动使用

所有的java虚拟机实现 必须在每个类或接口被java程序“首次主动使用”时才初始化它们

  • java程序的使用方式分为两种
  1. 主动使用(七种)
  • 创建类的实例

  • 访问某个类或接口的静态变量,或者对该静态变量赋值(助记符:getstatic,putstatic)

  • 调用类的静态方法(助记符:invokestatic)

  • 反射(如Calss.forName("com.test.Test"))

  • 初始化一个类的子类

初始化一个类的时候,如果该类有父类,也会对父类进行初始化,如果父类也有父类,也会往上进行初始化

  • Java虚拟机启动时被标记为启动类的类(Java Test)

  • JDK1.7开始提供的动态语言支持

java.lang.invoke.MethodHandle实例的解析结果 REF_getStatic, REF_putStatic,REF_invokeStatic句柄对应的类没有初始化,则初始化

  1. 被动使用
  • 除了上述其中情况,其他使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化

被动使用也许会加载这个类,只是不进行初始化

类的加载

  • 类的加载

类的加载是指将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在内存中创建一个java.lang.Class对象(规范并未说明Class对象位于哪里,HotSpot虚拟机将其放在方法区内)用来封装类在方法区内的数据结构

加载.class文件的方式

  1. 从本地文件系统中直接加载(最常用)
  2. 通过网络下载.class文件
  3. 从zip,jar等归档问价那种加载.calss文件
  4. 从专有的数据库中提取.calss文件
  5. 将Java源文件动态编译为.calss文件(动态代理,运行期创建,web开发会用到)

举例

  1. eg1
  2. public class MyTest {
  3. public static void main(String[] args) {
  4. System.out.println(MyChild1.str1);
  5. }
  6. }
  7. class MyParent1 {
  8. public static String str1 = "hello world str1";
  9. static {
  10. System.out.println("MyParent1 static block");
  11. }
  12. }
  13. class MyChild1 extends MyParent1 {
  14. public static String str2 = "hello world str2";
  15. static {
  16. System.out.println("MyChiled1 static block");
  17. }
  18. }
  19. //输出结果:
  20. MyParent1 static block
  21. hello world str1

对于静态字段来说,只有直接定义了该字段的类才会被初始化。所以在本例中,直接使用的是MyParent1的str1,所以会对MyParent1进行初始化,不会对MyChild1进行初始化,对MyParent1是主动使用

符合上述所有的java虚拟机实现必须在每个类或接口被java程序“首次主动使用”时才初始化它们

  1. eg2
  2. public class MyTest {
  3. public static void main(String[] args) {
  4. System.out.println(MyChild1.str2);
  5. }
  6. }
  7. class MyParent1 {
  8. public static String str1 = "hello world str1";
  9. static {
  10. System.out.println("MyParent1 static block");
  11. }
  12. }
  13. class MyChild1 extends MyParent1 {
  14. public static String str2 = "hello world str2";
  15. static {
  16. System.out.println("MyChiled1 static block");
  17. }
  18. }
  19. //输出结果:
  20. MyParent1 static block
  21. MyChiled1 static block
  22. hello world str2

初始化一个子类时,要求其父类全部已经初始化,在本例中,直接使用的是MyChild1的str2,所以会对MyChild1进行初始化,但是由上述的主动使用中初始化一个类的子类可知,对于父类也会进行初始化。(每个类只会被初始化一次)

所以本例中会先初始化MyParent1在初始化MyChild1,最后再执行输出MyChild1.str2操作

  1. eg3
  2. //执行主类的 VM arguments 中 添加 -XX:+TraceClassLoading
  3. //打印类加载信息
  4. public class MyTest {
  5. public static void main(String[] args) {
  6. System.out.println(MyChild1.str1);
  7. }
  8. }
  9. class MyParent1 {
  10. public static String str1 = "hello world str1";
  11. static {
  12. System.out.println("MyParent1 static block");
  13. }
  14. }
  15. class MyChild1 extends MyParent1 {
  16. public static String str2 = "hello world str2";
  17. static {
  18. System.out.println("MyChiled1 static block");
  19. }
  20. }
  21. //输出结果
  22. [Loaded java.lang.Object from shared objects file]
  23. [Loaded java.io.Serializable from shared objects file]
  24. ...
  25. [Loaded java.security.UnresolvedPermission from shared objects file]
  26. [Loaded java.security.BasicPermissionCollection from shared objects file]
  27. [Loaded cn.lillcol.classloader.MyTest from file:/E:/lteworkspace/gaSvr/target/classes/]
  28. [Loaded sun.launcher.LauncherHelper$FXHelper from shared objects file]
  29. [Loaded java.lang.Class$MethodArray from shared objects file]
  30. [Loaded java.lang.Void from shared objects file]
  31. [Loaded cn.lillcol.classloader.MyParent1 from file:/E:/lteworkspace/gaSvr/target/classes/]
  32. [Loaded cn.lillcol.classloader.MyChild1 from file:/E:/lteworkspace/gaSvr/target/classes/]
  33. MyParent1 static block
  34. hello world str1
  35. [Loaded java.lang.Shutdown from shared objects file]
  36. [Loaded java.lang.Shutdown$Lock from shared objects file]

加载的第一个类是java.lang.Object,它是所有类的父类

虽然 MyChild1 没有初始化,但是虚拟机还是加载了MyChild1

与我们编写的相关类加载顺序MyTest、MyParent1、MyChild1,其中MyTest类为Java虚拟机启动时被标记为启动类的类

  1. eg4:
  2. public class MyTest {
  3. static {
  4. System.out.println("MyTest static block");
  5. }
  6. public static void main(String[] args) {
  7. System.out.println(MyChild1.str1);
  8. }
  9. }
  10. class MyParent1 {
  11. public static String str1 = "hello world str1";
  12. static {
  13. System.out.println("MyParent1 static block");
  14. }
  15. }
  16. class MyChild1 extends MyParent1 {
  17. public static String str2 = "hello world str2";
  18. static {
  19. System.out.println("MyChiled1 static block");
  20. }
  21. }
  22. //输出结果:
  23. [Loaded cn.lillcol.classloader.MyTest from file:/E:/lteworkspace/gaSvr/target/classes/]
  24. [Loaded sun.launcher.LauncherHelper$FXHelper from shared objects file]
  25. [Loaded java.lang.Class$MethodArray from shared objects file]
  26. [Loaded java.lang.Void from shared objects file]
  27. MyTest static block
  28. [Loaded cn.lillcol.classloader.MyParent1 from file:/E:/lteworkspace/gaSvr/target/classes/]
  29. [Loaded cn.lillcol.classloader.MyChild1 from file:/E:/lteworkspace/gaSvr/target/classes/]
  30. MyParent1 static block
  31. hello world str1
  32. [Loaded java.lang.Shutdown from shared objects file]
  33. [Loaded java.lang.Shutdown$Lock from shared objects file]

MyTest 加载顺序在MyParent1、MyChild1之前,为Java虚拟机启动时被标记为启动类的类

  1. eg5
  2. public class MyTest2 {
  3. public static void main(String[] args) {
  4. System.out.println(MyParent2.str2);
  5. }
  6. }
  7. class MyParent2 {
  8. public static final String str2 = "hello world";
  9. static {
  10. System.out.println("MyParent2 static block");
  11. }
  12. }
  13. //输出结果
  14. hello world

常量在编译阶段,会被存入到调用这个常量的方法所在的类常量池中.

本质上,调用类并没有直接引用到定义常量的类,因此并不会触发定义常量的类初始化

注意:这里指的是将常量str2存放到MyTest2的常量池中,之后MyTest2与MyParent2就没有任何关系了,甚至此时将MyParent2的.class文件删除也没有关系

  1. eg6:
  2. public class MyTest3 {
  3. public static void main(String[] args) {
  4. System.out.println(MyParent3.str3);
  5. }
  6. }
  7. class MyParent3{
  8. public static final String str3= UUID.randomUUID().toString();
  9. static {
  10. System.out.println("MyParent3 static block ");
  11. }
  12. }
  13. //输出结果:
  14. MyParent3 static block
  15. 62322324-7b63-49c8-a8c4-f6607421f7ff

如果一个常量值不是在编译期间可以确定,那么其值就不会放到调用类的常量池中

这时在运行程序的时候,会导致主动使用这个常量所在的类,显然会导致这个类被初始化

  1. eg7:在终端下用 javap -c 命令反编译MyTest2的.calss文件
  2. javap -c MyTest2
  3. #输出结果
  4. Compiled from "MyTest2.java"
  5. public class cn.lillcol.classloader.MyTest2 {
  6. public cn.lillcol.classloader.MyTest2();
  7. Code:
  8. 0: aload_0
  9. 1: invokespecial #8 // Method java/lang/Object."<init>":()V
  10. 4: return
  11. public static void main(java.lang.String[]);
  12. Code:
  13. 0: getstatic #16 // Field java/lang/System.out:Ljava/io/PrintStream;
  14. 3: ldc #22 // String hello world
  15. 5: invokevirtual #24 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
  16. 8: return
  17. }

助记符:ldc 表示将int、float或是String类型的常量值从常量池中推送至栈顶

  1. eg8:将代码作如下修改
  2. public class MyTest2 {
  3. public static void main(String[] args) {
  4. System.out.println(MyParent2.s);
  5. }
  6. }
  7. class MyParent2 {
  8. public static final String str2 = "hello world";
  9. public static final short s = 7;
  10. static {
  11. System.out.println("MyParent2 static block");
  12. }
  13. }
  14. //输出:
  15. 7
  16. //反编译结果:
  17. Compiled from "MyTest2.java"
  18. public class cn.lillcol.classloader.MyTest2 {
  19. public cn.lillcol.classloader.MyTest2();
  20. Code:
  21. 0: aload_0
  22. 1: invokespecial #8 // Method java/lang/Object."<init>":()V
  23. 4: return
  24. public static void main(java.lang.String[]);
  25. Code:
  26. 0: getstatic #16 // Field java/lang/System.out:Ljava/io/PrintStream;
  27. 3: bipush 7
  28. 5: invokevirtual #22 // Method java/io/PrintStream.println:(I)V
  29. 8: return
  30. }

助记符:bipush 表示将单字节(-128 - 127)的常量值从常量池中推送至栈顶

  1. eg9:将代码作如下修改
  2. public class MyTest2 {
  3. public static void main(String[] args) {
  4. System.out.println(MyParent2.i);
  5. }
  6. }
  7. class MyParent2 {
  8. public static final String str2 = "hello world";
  9. public static final short s = 7;
  10. public static final int i = 128;
  11. static {
  12. System.out.println("MyParent2 static block");
  13. }
  14. }
  15. //输出:
  16. 128
  17. //反编译结果:
  18. Compiled from "MyTest2.java"
  19. public class cn.lillcol.classloader.MyTest2 {
  20. public cn.lillcol.classloader.MyTest2();
  21. Code:
  22. 0: aload_0
  23. 1: invokespecial #8 // Method java/lang/Object."<init>":()V
  24. 4: return
  25. public static void main(java.lang.String[]);
  26. Code:
  27. 0: getstatic #16 // Field java/lang/System.out:Ljava/io/PrintStream;
  28. 3: sipush 128
  29. 6: invokevirtual #22 // Method java/io/PrintStream.println:(I)V
  30. 9: return
  31. }

助记符:sipush 表示将单字节(-32768 - 32767)的常量值从常量池中推送至栈顶

  1. eg10:将代码作如下修改
  2. public class MyTest2 {
  3. public static void main(String[] args) {
  4. System.out.println(MyParent2.m);
  5. }
  6. }
  7. class MyParent2 {
  8. public static final String str2 = "hello world";
  9. public static final short s = 7;
  10. public static final int i = 128;
  11. public static final int m = 0;
  12. static {
  13. System.out.println("MyParent2 static block");
  14. }
  15. }
  16. //输出:
  17. 0
  18. //反编译结果:
  19. Compiled from "MyTest2.java"
  20. public class cn.lillcol.classloader.MyTest2 {
  21. public cn.lillcol.classloader.MyTest2();
  22. Code:
  23. 0: aload_0
  24. 1: invokespecial #8 // Method java/lang/Object."<init>":()V
  25. 4: return
  26. public static void main(java.lang.String[]);
  27. Code:
  28. 0: getstatic #16 // Field java/lang/System.out:Ljava/io/PrintStream;
  29. 3: iconst_0
  30. 4: invokevirtual #22 // Method java/io/PrintStream.println:(I)V
  31. 7: return
  32. }

助记符:iconst_0 表示将int类型1从常量池中推送至栈顶(iconst_m1 - iconst_5)

助记符存储的位置:jdk的rt.jar中

  1. eg11:
  2. public class MyTest4 {
  3. public static void main(String[] args) {
  4. MyParent4[] myTest4s = new MyParent4[1];
  5. System.out.println(myTest4s.getClass());
  6. MyParent4[][] myTest4s1 = new MyParent4[1][1];
  7. System.out.println(myTest4s1.getClass());
  8. System.out.println(myTest4s.getClass().getSuperclass());
  9. System.out.println(myTest4s1.getClass().getSuperclass());
  10. int[] ints=new int[1];
  11. System.out.println(ints.getClass());
  12. System.out.println(ints.getClass().getSuperclass());
  13. }
  14. }
  15. class MyParent4{
  16. static {
  17. System.out.println("MyParent4 static block");
  18. }
  19. }
  20. //输出:
  21. class [Lcom.aaa.test.MyParent4;
  22. class [[Lcom.aaa.test.MyParent4;
  23. class java.lang.Object
  24. class java.lang.Object
  25. class [I
  26. class java.lang.Object

对于数组实例来说,其类型是由JVM在运行期间动态生成的,表示为class [Lcom.aaa.test.MyParent4;这种形式,动态生成的类型其父类是java.lang.Object。

对于数组来说,JavaDoc经常将构成数组的元素称为Component,实际上就是将数组降低一个维度后的类型。

  1. eg12
  2. int[] ints=new int[1];
  3. System.out.println(ints.getClass());
  4. short[] showts=new short[1];
  5. System.out.println(showts.getClass());
  6. boolean[] booleans=new boolean[1];
  7. System.out.println(booleans.getClass());
  8. char[] chars=new char[1];
  9. System.out.println(chars.getClass());
  10. byte[] bytes=new byte[1];
  11. System.out.println(bytes.getClass());
  12. float[] floats=new float[1];
  13. System.out.println(floats.getClass());
  14. double[] doubles=new double[1];
  15. System.out.println(doubles.getClass());
  16. //输出:
  17. class [I
  18. class [S
  19. class [Z
  20. class [C
  21. class [B
  22. class [F
  23. class [D

助记符:

anewarray: 表示创建一个引用类型(如类、接口、数组)数组,并将其引用值压入栈顶

ewarray:表示创建一个指定的原始类型(如int、float、char等)的数组,并将其引用值压入栈顶

JVM参数

  1. -XX:+<option>,表示开启option选项
  2. -XX:-<option>,表示关闭option选项
  3. -XX:<option>=<value>,表示将option选项的值设置value

+、-是因为有些参数默认开启或者关闭

本文为学习张龙老师深入理解JVM的笔记与心得,转载请注明出处!!!

深入理解JVM(一)类加载器部分、类变量、常量、jvm参数的更多相关文章

  1. 【深入理解JVM】类加载器与双亲委派模型 (转)

    出处: [深入理解JVM]类加载器与双亲委派模型 加载类的开放性 类加载器(ClassLoader)是Java语言的一项创新,也是Java流行的一个重要原因.在类加载的第一阶段“加载”过程中,需要通过 ...

  2. JVM自定义类加载器加载指定classPath下的所有class及jar

    一.JVM中的类加载器类型 从Java虚拟机的角度讲,只有两种不同的类加载器:启动类加载器和其他类加载器. 1.启动类加载器(Boostrap ClassLoader):这个是由c++实现的,主要负责 ...

  3. Java JVM——2.类加载器子系统

    概述 类加载器子系统在Java JVM中的位置 类加载器子系统的具体实现 类加载器子系统的作用 ① 负责从文件系统或者网络中加载.class文件,Class 文件在文件开头有特定的文件标识. ② Cl ...

  4. 1. JVM核心类加载器及类加载的全过程

    运行环境: 下面说明一下我的运行环境.我是在mac上操作的. 先找到mac的java地址. 从~/.bash_profile中可以看到 java的home目录是: /Library/Java/Java ...

  5. 1.1 jvm核心类加载器--jdk源码剖析

    目录 前提: 运行环境 1. 类加载的过程 1.1 类加载器初始化的过程 1.2 类加载的过程 1.3 类的懒加载 2. jvm核心类加载器 3. 双亲委派机制 4. 自定义类加载器 5. tomca ...

  6. 深入理解:java类加载器

    概念理解:Java类加载器总结 1.深入理解Java类加载器(1):Java类加载原理解析 2.深入理解Java类加载器(2):线程上下文类加载器

  7. JVM 自定义类加载器

    一.创建自定义类加载器 package com.example.jvm.classloader; import java.io.ByteArrayOutputStream; import java.i ...

  8. 【深入理解JVM】类加载器与双亲委派模型

    原文链接:http://blog.csdn.net/u011080472/article/details/51332866,http://www.cnblogs.com/lanxuezaipiao/p ...

  9. 深入理解JVM一类加载器原理

    我们知道我们编写的java代码,会经过编译器编译成字节码文件(class文件),再把字节码文件装载到JVM中,映射到各个内存区域中,我们的程序就可以在内存中运行了.那么字节码文件是怎样装载到JVM中的 ...

  10. 深入理解JVM,类加载器

    虚拟机设计团队把类加载阶段中的“通过一个类的全限定名来获取描述此类的二进制字节流(即字节码)”这个动作放到Java虚拟机外部去实现,以便让应用程序自己决定如何去获取所需要的类.实现这个动作的代码模块称 ...

随机推荐

  1. HDU 3308 线段树求区间最长连续上升子序列长度

    题意:两种操作,Q L R查询L - R 的最长连续上升子序列长度,U pos val 单点修改值 #include <bits/stdc++.h> #define N 100005 us ...

  2. 针对Java集合类的小总结

    Java集合类包位于java.util下,有很多常用的数据结构:数组.链表.队列.栈.哈希表等等.了解不同的集合类的特性在开发过程中是比较重要的,感谢@兰亭风雨的专栏分析,这里我也根据自己的理解做轻度 ...

  3. java 3des加密问题记录

    3des加密有不同的加密模式和填充模式,这个网上很多不多说了,只要保证加解密的时候加密模式和填充模式保持一致就可以了 首先对于密钥的生成,java中有2种方式: 1.第一种,采用ECB模式和不填充模式 ...

  4. The linux command 之网络

    一.检查和检测网络 ping命令——向网络主机发送特殊数据包 [me@linuxbox ~]$ ping www.baidu.com 按Ctrl+C终止程序 tracepath——跟踪网络数据包的传输 ...

  5. Delphi 第2课

    项目文件.dpr单元文件.pas--目标文件.dcu窗体文件.dfu begin 对象名.属性 对象名.方法end; 在delphi 自动提示控制语句结构 键盘输入Ctrl 就可以看到了 代码错误:( ...

  6. 【JZOJ6350】考试(test)

    description analysis 对于\(n=0\)的点,直接模拟就好了 状压\(DP\),设\(f[i][j][S]\)表示到第\(i\)题.连续\(GG\)了\(j\)题.喝的饮料集合为\ ...

  7. 20170702-变量说明,静态方法,类方法区别,断点调试,fork,yield协程,进程,动态添加属性等。。

    概念: 并行:同时运行 并发:看似同时运行  json后任然中文的问题 import json d = {"名字":"初恋这件小事"} new_d1 = jso ...

  8. 大数据之hadoop集群安全模式

    集群安全模式1.概述(1)NameNode启动 NameNode启动时,首先将镜像文件(Fsimage)载入内存,并执行编辑日志(Edits)中的各项操作.-旦在内存中成功建立文件系统元数据的影像,则 ...

  9. 学习js和jQuery总结

    数据类型的转换:1.字符串必须放在引号内2.字符串的转换有两个标签,string和tostring这两个函数.3.变量必须使用驼峰形式,把一个变量转换成字符串:console.log(isChild. ...

  10. 修改 ulimit 参数

    sudo vim /etc/security/limits.conf 文件下加: * hard nofile 999999 * soft nofile 999999 * soft nproc 1024 ...