A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

本帖最后由 我爱睡觉 于 2016-1-20 16:30 编辑

原文出处: iceAeterna 
Java虚拟机通过装载、连接、初始化来使得一个Java类型可以被Java程序所使用,如下图所示,其中连接过程又分为验证、准备、解析三个部分。其中部分类的解析过程可以推迟到程序真正使用其某个符号引用时再去解析。

解析过程可以推迟到类的初始化之后再进行,但这是有条件的,Java虚拟机必须在每个类或接口主动使用时进行初始化。
以下为主动使用的情况:
(1).(无论直接通过new创建出来的,还是通过反射、克隆、序列化创建的)创建某个类新的实例
(2).使用某个类的静态方法
(3).访问某个类或接口的静态字段
(4).调用JavaAPI中的某些反射方法
(5).初始化某个类的子类(要求其祖先类都要被初始化,否则无法正确访问其继承的成员)
(6).启动某个标明为启动类的类(含有main()方法)
主动使用会导致类的初始化,其超类均将在该类的初始化之前被初始化,但通过子类访问父类的静态字段或方法时,对于子类(或子接口、接口的实现类)来说,这种访问就是被动访问,或者说访问了该类(接口)中的不在该类(接口)中声明的静态成员。
如:
Grandpa的定义如下:
  1. package com.ice.passiveaccess;

  2. public class Grandpa {
  3.     static{
  4.         System.out.println("Grandpa was initialized.");
  5.     }
  6. }
复制代码






Parent的定义如下:
  1. package com.ice.passiveaccess;

  2. public class Parent extends Grandpa{
  3.     static String language = "Chinese";
  4.     static{
  5.         System.out.println("Parent was initialized.");
  6.     }
  7. }
复制代码


Cindy的定义如下:
  1. package com.ice.passiveaccess;

  2. public class Cindy extends Parent{
  3.     static{
  4.         System.out.println("Child was initialized.");
  5.     }
  6. }
复制代码



现在通过Cindy访问父类的language成员
  1. package com.ice.passiveaccess;

  2. public class PassiveAccessTest {
  3.     public static void main(String args[]){
  4.         System.out.println(Cindy.language);
  5.     }
  6. }
复制代码



结果如下:

可见这是被动访问,Cindy自身并没有初始化
下面简要介绍装载、验证与初始化过程:
1.装载:
(1).找到该类型的class文件,产生一个该类型的class文件二进制数据流(ClassLoader需要实现的loadClassData()方法)
(2).解析该二进制数据流为方法区内的数据结构
(3).创建一个该类型的java.lang.Class实例
在加载器的相关代码中可以看到,最终通过defineClass()创建一个Java类型对象(Class对象)。
2.验证:
class文件校验器需要四趟独立的扫描来完成验证工作,其中:
第一趟扫描在装载时进行,会对class文件进行结构检查,如
(1).对魔数进行检查,以判断该文件是否是一个正常的class文件
(2).对主次版本号进行检查,以判断class文件是否与java虚拟机兼容
(3).对class文件的长度和类型进行检查,避免class文件部分缺失或被附加内容。
第二趟扫描在连接过程中进行,会对类型数据进行语义检查,主要检查各个类的二进制兼容性(主要是查看超类和子类的关系)和类本身是否符合特定的语义条件
(1).final类不能拥有子类
(2).final方法不能被重写(覆盖)
(3).子类和超类之间没有不兼容的方法声明
(4).检查常量池入口类型是否一致(如CONSTANT_Class常量池的内容是否指向一个CONSTANT_Utf8字符串常量池)
(5).检查常量池的所有特殊字符串,以确定它们是否是其所属类型的实例,以及是否符合特定的上下文无关语法、格式
第三趟扫描为字节码验证,其验证内容和实现较为复杂,主要检验字节码是否可以被java虚拟机安全地执行。
第四趟扫描在解析过程中进行,为对符号引用的验证。在动态连接过程中,通过保存在常量池的符号引用查找被引用的类、接口、字段、方法时,在把符号引用替换成直接引用时,首先需要确认查找的元素真正存在,然后需要检查访问权限、查找的元素是否是静态类成员而非实例成员。
3.准备:
为类变量分配内存、设置默认初始值(内存设置初始值,而非对类变量真正地进行初始化,即类中声明int i = 5,但实际上这里是分配内存并设置初始值为0)
4.解析:
在类的常量池中寻找类、接口、字段、方法的符号引用,将这些符号引用替换成直接引用
5.初始化:
对类变量赋予指定的初始值(这个时候int i = 5就必须赋予i以初值5)。这个初始值的给定方式有两种,一种是通过类变量的初始化语句,一种是静态初始化语句。而这些初始化语句都将被Java编译器一起放在方法中。
如前面所述,一个类的初始化需要初始化其直接超类,并递归初始化其祖先类,初始化是通过调用类的初始化方法完成的。此外,对于接口,并不需要初始化其父接口,而只需要执行该接口的接口初始化方法就可以了。
注意:
(1). 在初始化阶段,只会为类变量(静态全局变量)进行初始化工作,并且当类变量声明为final类型切初始化语句采用了常量表达式方式进行初始化赋值,那么, 也不会对其进行初始化,它将会直接被编译器计算并保存在常量池中,并且对这些变量的使用也将直接将其变量值嵌入到字节码中。
如UsefulParameter类如下:
  1. Class UsefulParameter{
  2. static final int height = 2;
  3. static final int width = height * 2;
  4. }
复制代码




类Area的类变量初始化如下:
  1. Class Area{
  2. static int height = UsefulParameter.height * 2 ;
  3. static int width = UsefulParameter.width * 2;
  4. }
复制代码




在Area的< clinit>中,将直接把2、4嵌入到字节码中

(2).接口的初始化与类有所不同,在初始化阶段,会为在接口中声明的所有public、static和final类型的、无法被编译为常量的字段进行初始化








0 个回复

您需要登录后才可以回帖 登录 | 加入黑马