01类加载子系统
一、类加载器子系统作用
类加载器子系统负责从文件系统或者网络中加载 class 文件,class 文件在文件开头有特定的文件标识。
ClassLoader 只负责 class 文件的加载,至于它是否可以运行,则由 Execution Engine 决定。
加载的类信息存放于一块称为方法区的内存空间。除了类的信息外,方法区中还会存放运行时常量池信息,可能还包括字符串字面量和数字常量(这部分常量信息是 class 文件中常量池部分的内存映射)
二、类加载器ClasLoader角色
class file 存在于本地硬盘上,可以理解为设计师画在纸上的模板,而最终这个模板在执行的时候是要加载到JVM 当中来根据这个文件实例化出 n 个一模一样的实例。
class file 加载到 JVM 中,被称为 DNA 元数据模板,放在方法区。
在 class 文件 -> JVM -> 最终成为元数据模板,此过程就要一个运输工具(类装载器Class Loader),扮演一个快递员的角色。
三、类的加载过程
public class HelloLoader { public static void main(String[] args) { System.out.println("Hello World!"); } }
四、加载阶段
通过一个类的全限定名获取定义此类的二进制字节流
将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构
在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口
加载class文件的方式
从本地系统中直接加载
通过网络获取,典型场景:Web Applet
从zip压缩包中读取,成为日后jar、war格式的基础
运行时计算生成,使用最多的是:动态代理技术
由其他文件生成,典型场景:JSP应用
从专有数据库中提取.class文件,比较少见
从加密文件中获取,典型的防Class文件被反编译的保护措施
五、链接阶段
1、验证(Verify)
目的在子确保Class文件的字节流中包含信息符合当前虚拟机要求,保证被加载类的正确性,不会危害虚拟机自身安全。
主要包括四种验证,文件格式验证,元数据验证,字节码验证,符号引用验证。
2、准备(Prepare)
为类变量分配内存并且设置该类变量的默认初始值,即零值。
这里不包含用final修饰的static,因为final在编译的时候就会分配了,准备阶段会显式初始化;
这里不会为实例变量分配初始化,类变量会分配在方法区中,而实例变量是会随着对象一起分配到Java堆中。
3、解析(Resolve)
将常量池内的符号引用转换为直接引用的过程。
事实上,解析操作往往会伴随着JVM在执行完初始化之后再执行。
符号引用就是一组符号来描述所引用的目标。符号引用的字面量形式明确定义在《java虚拟机规范》的Class文件格式中。直接引用就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄。
解析动作主要针对类或接口、字段、类方法、接口方法、方法类型等。对应常量池中的CONSTANT_Class_info,CONSTANT_Fieldref_info、CONSTANT_Methodref_info等。
六、初始化阶段
初始化阶段就是执行类构造器方法()的过程。
此方法不需定义,是javac编译器自动收集类中的所有类变量的赋值动作和静态代码块中的语句合并而来。如果没有这些时不会有clinit方法
构造器方法中指令按语句在源文件中出现的顺序执行。
()不同于类的构造器。(关联:构造器是虚拟机视角下的())
若该类具有父类,JVM会保证子类的()执行前,父类的()已经执行完毕。
虚拟机必须保证一个类的()方法在多线程下被同步加锁。
七、虚拟机加载器
1、启动类加载器(引导类加载器,Bootstrap ClassLoader)
这个类加载使用C/C++语言实现的,嵌套在JVM内部。
它用来加载Java的核心库(JAVA_HOME/jre/lib/rt.jar、resources.jar或sun.boot.class.path路径下的内容),用于提供JVM自身需要的类,并不继承自ava.lang.ClassLoader,没有父加载器。加载扩展类和应用程序类加载器,并指定为他们的父类加载器。
出于安全考虑,Bootstrap启动类加载器只加载包名为java、javax、sun等开头的类
具体加载
2、扩展类加载器(Extension ClassLoader)
Java语言编写,由sun.misc.Launcher$ExtClassLoader实现。
父类加载器为启动类加载器
从java.ext.dirs系统属性所指定的目录中加载类库,或从JDK的安装目录的jre/1ib/ext子目录(扩展目录)下加载类库。如果用户创建的JAR放在此目录下,也会自动由扩展类加载器加载。
具体加载下面位置类
3、应用程序类加载器(系统类加载器,AppClassLoader)
java语言编写,由sun.misc.LaunchersAppClassLoader实现
父类加载器为扩展类加载器
它负责加载环境变量classpath或系统属性java.class.path指定路径下的类库
该类加载是程序中默认的类加载器,一般来说,Java应用的类都是由它来完成加载
通过ClassLoader#getSystemclassLoader() 方法可以获取到该类加载器
八、用户自定义类加载器
在Java的日常应用程序开发中,类的加载几乎是由上述3种类加载器相互配合执行的,在必要时,我们还可以自定义类加载器,来定制类的加载方式。
● 隔离加载类
● 修改类加载的方式
● 扩展加载源
● 防止源码泄漏
用户自定义类加载器实现步骤:
1. 继承抽象类ava.lang.ClassLoader类的方式,实现自己的类加载器,以满足一些特殊的需求
2. 在JDK1.2之前,在自定义类加载器时,总会去继承ClassLoader类并重写loadClass() 方法,从而实现自定义的类加载类,但是在JDK1.2之后已不再建议用户去覆盖loadclass() 方法,而是建议把自定义的类加载逻辑写在findClass()方法中
3. 在编写自定义类加载器时,如果没有太过于复杂的需求,可以直接继承URLClassLoader 类,这样就可以避免自己去编写findClass() 方法及其获取字节码流的方式,使自定义类加载器编写更加简洁。
九、双亲委派机制
Java虚拟机对class文件采用的是按需加载的方式,也就是说当需要使用该类时才会将它的class文件加载到内存生成class对象 。而且加载某个类的class文件时,Java虚拟机采用的是双亲委派模式,即把请求交由父类处理,它是一种任务委派模式。
1、工作原理
1)如果一个类加载器收到了类加载请求,它并不会自己先去加载,而是把这个请求委托给父类的加载器去执行;
2)如果父类加载器还存在其父类加载器,则进一步向上委托,依次递归,请求最终将到达顶层的启动类加载器;
3)如果父类加载器可以完成类加载任务,就成功返回,倘若父类加载器无法完成此加载任务,子加载器才会尝试自己去加载,这就是双亲委派模式。
2、双亲委派举例
当我们加载jdbc.jar 用于实现数据库连接的时候,首先我们需要知道的是 jdbc.jar是基于SPI接口进行实现的,所以在加载的时候,会进行双亲委派,最终从根加载器中加载 SPI核心类,然后在加载SPI接口类,接着在进行反向委派,通过线程上下文类加载器进行实现类jdbc.jar的加载。
3、优势
避免类的重复加载
保护程序安全,防止核心API被随意篡改
○ 自定义类:java.lang.String
○ 自定义类:java.lang.ShkStart(报错:阻止创建 java.lang开头的类)
十、对类加载器的引用
JVM必须知道一个类型是由启动加载器加载的还是由用户类加载器加载的。如果一个类型是由用户类加载器加载的,那么JVM会将这个类加载器的一个引用作为类型信息的一部分保存在方法区中。当解析一个类型到另一个类型的引用的时候,JVM需要保证这两个类型的类加载器是相同的 。
十一、类的主动使用和被动使用
主动使用
● 创建类的实例
● 访问某个类或接口的静态变量,或者对该静态变量赋值
● 调用类的静态方法
● 反射(比如:Class.forName(""))
● 初始化一个类的子类
● Java虚拟机启动时被标明为启动类的类
● JDK 7 开始提供的动态语言支持:
java.lang.invoke.MethodHandle实例的解析结果
REF_getStatic、REF_putStatic、REF_invokeStatic句柄对应的类没有初始化,则初始化
除了以上七种情况,其他使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化 。
附:
对给定的class文件提供的字节代码进行反编译
javap -v Test01ClassInit.class
Classfile /E:/code/01base/00JVM/target/classes/com/ghbs/jvm01/Test01ClassInit.class
Last modified 2022-10-16; size 668 bytes
MD5 checksum 0e3834313ef891c744b4a55cf1ef51b8
Compiled from "Test01ClassInit.java"
public class com.ghbs.jvm01.Test01ClassInit
minor version: 0
major version: 52
flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
#1 = Methodref #6.#24 // java/lang/Object."":()V
#2 = Fieldref #25.#26 // java/lang/System.out:Ljava/io/PrintStream;
#3 = Fieldref #5.#27 // com/ghbs/jvm01/Test01ClassInit.number:I
#4 = Methodref #28.#29 // java/io/PrintStream.println:(I)V
#5 = Class #30 // com/ghbs/jvm01/Test01ClassInit
#6 = Class #31 // java/lang/Object
#7 = Utf8 number
#8 = Utf8 I
#9 = Utf8
#10 = Utf8 ()V
#11 = Utf8 Code
#12 = Utf8 LineNumberTable
#13 = Utf8 LocalVariableTable
#14 = Utf8 this
#15 = Utf8 Lcom/ghbs/jvm01/Test01ClassInit;
#16 = Utf8 main
#17 = Utf8 ([Ljava/lang/String;)V
#18 = Utf8 args
#19 = Utf8 [Ljava/lang/String;
#20 = Utf8 MethodParameters
#21 = Utf8
#22 = Utf8 SourceFile
#23 = Utf8 Test01ClassInit.java
#24 = NameAndType #9:#10 // "":()V
#25 = Class #32 // java/lang/System
#26 = NameAndType #33:#34 // out:Ljava/io/PrintStream;
#27 = NameAndType #7:#8 // number:I
#28 = Class #35 // java/io/PrintStream
#29 = NameAndType #36:#37 // println:(I)V
#30 = Utf8 com/ghbs/jvm01/Test01ClassInit
#31 = Utf8 java/lang/Object
#32 = Utf8 java/lang/System
#33 = Utf8 out
#34 = Utf8 Ljava/io/PrintStream;
#35 = Utf8 java/io/PrintStream
#36 = Utf8 println
#37 = Utf8 (I)V
{
public com.ghbs.jvm01.Test01ClassInit();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=1, locals=1, args_size=1
0: aload_0
1: invokespecial #1 // Method java/lang/Object."":()V
4: return
LineNumberTable:
line 6: 0
LocalVariableTable:
Start Length Slot Name Signature
0 5 0 this Lcom/ghbs/jvm01/Test01ClassInit;
public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=2, locals=1, args_size=1
0: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
3: getstatic #3 // Field number:I
6: invokevirtual #4 // Method java/io/PrintStream.println:(I)V
9: return
LineNumberTable:
line 14: 0
line 15: 9
LocalVariableTable:
Start Length Slot Name Signature
0 10 0 args [Ljava/lang/String;
MethodParameters:
Name Flags
args
static {};
descriptor: ()V
flags: ACC_STATIC
Code:
stack=1, locals=0, args_size=0
0: iconst_2
1: putstatic #3 // Field number:I
4: iconst_1
5: putstatic #3 // Field number:I
8: return
LineNumberTable:
line 8: 0
line 11: 4
}
SourceFile: "Test01ClassInit.java"
E:code 1base 0JVM argetclassescomghbsjvm01>
E:code 1base 0JVM argetclassescomghbsjvm01>
E:code 1base 0JVM argetclassescomghbsjvm01>
E:code 1base 0JVM argetclassescomghbsjvm01>javap -v Test01ClassInit.class
Classfile /E:/code/01base/00JVM/target/classes/com/ghbs/jvm01/Test01ClassInit.class
Last modified 2022-10-16; size 671 bytes
MD5 checksum 40c1efe2199adbecd985780dd2eac816
Compiled from "Test01ClassInit.java"
public class com.ghbs.jvm01.Test01ClassInit
minor version: 0
major version: 52
flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
#1 = Methodref #6.#24 // java/lang/Object."":()V
#2 = Fieldref #25.#26 // java/lang/System.out:Ljava/io/PrintStream;
#3 = Fieldref #5.#27 // com/ghbs/jvm01/Test01ClassInit.number:I
#4 = Methodref #28.#29 // java/io/PrintStream.println:(I)V
#5 = Class #30 // com/ghbs/jvm01/Test01ClassInit
#6 = Class #31 // java/lang/Object
#7 = Utf8 number
#8 = Utf8 I
#9 = Utf8
#10 = Utf8 ()V
#11 = Utf8 Code
#12 = Utf8 LineNumberTable
#13 = Utf8 LocalVariableTable
#14 = Utf8 this
#15 = Utf8 Lcom/ghbs/jvm01/Test01ClassInit;
#16 = Utf8 main
#17 = Utf8 ([Ljava/lang/String;)V
#18 = Utf8 args
#19 = Utf8 [Ljava/lang/String;
#20 = Utf8 MethodParameters
#21 = Utf8
#22 = Utf8 SourceFile
#23 = Utf8 Test01ClassInit.java
#24 = NameAndType #9:#10 // "":()V
#25 = Class #32 // java/lang/System
#26 = NameAndType #33:#34 // out:Ljava/io/PrintStream;
#27 = NameAndType #7:#8 // number:I
#28 = Class #35 // java/io/PrintStream
#29 = NameAndType #36:#37 // println:(I)V
#30 = Utf8 com/ghbs/jvm01/Test01ClassInit
#31 = Utf8 java/lang/Object
#32 = Utf8 java/lang/System
#33 = Utf8 out
#34 = Utf8 Ljava/io/PrintStream;
#35 = Utf8 java/io/PrintStream
#36 = Utf8 println
#37 = Utf8 (I)V
{
public com.ghbs.jvm01.Test01ClassInit();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=1, locals=1, args_size=1
0: aload_0
1: invokespecial #1 // Method java/lang/Object."":()V
4: return
LineNumberTable:
line 6: 0
LocalVariableTable:
Start Length Slot Name Signature
0 5 0 this Lcom/ghbs/jvm01/Test01ClassInit;
public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=2, locals=1, args_size=1
0: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
3: getstatic #3 // Field number:I
6: invokevirtual #4 // Method java/io/PrintStream.println:(I)V
9: return
LineNumberTable:
line 14: 0
line 15: 9
LocalVariableTable:
Start Length Slot Name Signature
0 10 0 args [Ljava/lang/String;
MethodParameters:
Name Flags
args
static {};
descriptor: ()V
flags: ACC_STATIC
Code:
stack=1, locals=0, args_size=0
0: sipush 200
3: putstatic #3 // Field number:I
6: bipush 100
8: putstatic #3 // Field number:I
11: return
LineNumberTable:
line 8: 0
line 11: 6
}
SourceFile: "Test01ClassInit.java"
一个会计的水平能差到什么程度?不计提固定资产折旧,不帮员工代扣个税。账一塌糊涂。我们医院的前任会计,因为跟老板吵架,一气之下丢下工作不来了。然后现任会计被请过来帮忙。首先,清理旧账。因为前任会计不肯交接,只留下
肺癌晚期,吃靶向药有没有患者不耐药的?很遗憾,肿瘤靶向治疗最后都会出现耐药,难以避免,肺癌也不例外,这也是医学界面临的最大难题之一。换句话说,如果肿瘤对靶向药物一直敏感,药物一直都有效,也就意味着肿瘤治愈了,或者就像高
炒股不盯盘可以吗?有两个美女朋友。两个人资金量都差不多50多万。一个喜欢做差价,天天非常勤快,每天不停的折腾,天天倒腾个几百一千,自己感觉其乐无穷,不管股票亏赚反正就是一直做T。另一个懒得要死,不管
要不要逼小孩去运动?千万不要逼!要好好引导!我家孩子两三岁的时候有一次家人带出去玩的时候可能是被别人推了还是怎样,我再带出去就不敢玩了,总粘在我身边看别人玩。然后我是牵着她们手一起玩,刚开始找人少或没
一个月给你八千,叫你上十二休十二,没有假期,你能坚持多久,你乐意吗?上12个小时,休12个小时,也就是两班倒,但没有假期,实际上目前有很多公司存在这样的岗位,但一个月完全没有假期的比较少,一般一个月得休息个4天,不然长期下去,人受不了。如果上12小
将钱放余额宝安全吗?朋友们好,说起余额宝,与许多理财朋友都息息相关,利益相连。非常明确的回复将钱放,余额宝,二级低风险,安全性较为突出,适用人群极为广泛。但也有需要深入了解的地方,知己知彼,投资理财更
文科生如果想要高考过六百分,需要怎么安排各科的学习?我觉得你这几科除了数学稍微弱一点,其他的还都挺平均的,所以你现在得试试把五十分的任务摊给这几科。首先是数学,数学100分很拖后腿,我的建议是先把数学分数提高到120分,选择题和填空
脑萎缩吃什么好?脑萎缩的形成,按照我的经验和理解,就是脑部的营养缺乏,脑组织处于慢性缺血缺氧状态,脑细胞功能受到影响,使细胞数目减少,体积逐渐缩小,即形成脑萎缩,无法控制神经,况且神经也需要养分的
DNF中的红眼多少独立才算及格?首先,DNF红眼属性及格线独立2500,力量3200,属强200。但个人观点,装备有了,武器锻造8就行!其次,普及一下有哪些方式堆独立攻击1武器锻造,目前最高级别是锻895版本圣耀
为什么塞尔达传说中文版叫塞尔达传说,旷野之息而不是翻译过来一直叫的塞尔达传说,荒野之息?英文名字Breathofthewild中,wild的原版英文释义是anaturalstateoruncultivatedoruninhabitedregion,翻成中文就是一个自然
王艺迪上周还输给58岁倪夏莲,现在怎么连赢早田希娜和伊藤美诚?本月上旬进行的wtt挑战赛,王艺迪输给了58岁的倪夏莲,爆出了大冷门。下旬进行的世乒赛,王艺迪发挥极其亮眼,连胜伊藤美诚早田希娜两位新星,成功担起来扫雷重任,帮助国乒包揽四强,提前