JVM学习笔记(四):类的初始化
2015-04-22 21:56
232 查看
最近一直忙着面试,一直没有时间写这篇类的初始化。被阿里虐完,加上实验室比较吵,不想看新知识,还是把这篇文章写了吧。
首先要明确这个标题,是类的初始化,而不是对象的初始化,他们千差万别不要搞混。
类初始化:就是按照程序员的意愿执行类的初始化(clinit)方法,给静态变量等赋初值(之前提到准备阶段是给类的的静态变量分配内存,赋null或0值等)。
类的初始化语句:静态变量声明语句,以及静态代码块。
类的主动使用:
创建类的实例、
访问类或接口的静态变量,或者对静态变量赋值的,或者调用类的静态方法
使用反射
初始化类的子类
JVM启动时的启动类(main方法)
类的初始化时机:只有主动使用类的时候才会导致初始化。
JVM初始化一个类的时候会要求其所有父类已经被初始化,但是不适用与接口,父接口不会因为子接口或者实现类的初始化而被初始化。
只有程序访问静态变量或静态方法在当前类的时候,才可以认为是当前类的主动使用
调用ClassLoader的loadClass方法加载一个类的时候,不是类的主动使用
main block output
parent block
10
10
虽然用Child调用的a,但是可以发现Child未被初始化,因为第二点所说,a并不是Child中的变量,而是Parent中的
初始化步骤:
如果类未被加载连接,则想进行加载连接
如果类的直接父类未被初始化,先初始化父类
类中有初始化语句(静态变量或静态块),则执行初始化语句
此时的输出结果是多少呢:
先不说答案,先分析。
在类的准备阶段,会给Parent的静态变量p,n,m分配内存,并赋初始值,p赋值为null,n,m赋值为0;
程序执行Parent.getInstance()的时候是对类的主动使用,将导致类的初始化,Parent类中按照程序员的意愿给静态变量赋初值,调用Parent的构造方法,将n,m各自加1,这是n=1,m=1。然后按照顺序执行static int n;这时n的值并没有改变,还是1;然后执行static int m=n+5;这时m的值被被修改,变为6。所有程序的输出为n=1,m=6。
首先要明确这个标题,是类的初始化,而不是对象的初始化,他们千差万别不要搞混。
类初始化:就是按照程序员的意愿执行类的初始化(clinit)方法,给静态变量等赋初值(之前提到准备阶段是给类的的静态变量分配内存,赋null或0值等)。
类的初始化语句:静态变量声明语句,以及静态代码块。
类的主动使用:
创建类的实例、
访问类或接口的静态变量,或者对静态变量赋值的,或者调用类的静态方法
使用反射
初始化类的子类
JVM启动时的启动类(main方法)
类的初始化时机:只有主动使用类的时候才会导致初始化。
JVM初始化一个类的时候会要求其所有父类已经被初始化,但是不适用与接口,父接口不会因为子接口或者实现类的初始化而被初始化。
只有程序访问静态变量或静态方法在当前类的时候,才可以认为是当前类的主动使用
调用ClassLoader的loadClass方法加载一个类的时候,不是类的主动使用
class Parent { static int a; static { System.out.println("parent block"); a = 10; System.out.println(a); } } class Child extends Parent { static int b; static { System.out.println("child block"); b = 11; System.out.println(b); } } public class InitParent { public static void main(String[] args) { System.out.println("main block output"); System.out.println(Child.a); } }输出:
main block output
parent block
10
10
虽然用Child调用的a,但是可以发现Child未被初始化,因为第二点所说,a并不是Child中的变量,而是Parent中的
初始化步骤:
如果类未被加载连接,则想进行加载连接
如果类的直接父类未被初始化,先初始化父类
类中有初始化语句(静态变量或静态块),则执行初始化语句
class Parent { static int a; static { System.out.println("parent block"); a = 10; System.out.println(a); } } class Child extends Parent { static int b; static { System.out.println("child block"); b = 11; System.out.println(b); } } public class InitParent { public static void main(String[] args) { Child c = new Child(); System.out.println("main block output"); System.out.println(c.b); } }
//输出结果如下 parent block 10 child block 11 main block output 11
//可以很容易发现,在初始化子类之前初始化了父类。
//下面是一个很恶心的笔试题: <pre name="code" class="java">class Parent { static Parent p = new Parent(); static int n; static int m = n + 5; public Parent() { n++; m++; } public static Parent getInstance() { return p; } } public class InitParent { public static void main(String[] args) { Parent p = Parent.getInstance(); System.out.println("n = " + Parent.n); System.out.println("m = " + p.m); } }
此时的输出结果是多少呢:
先不说答案,先分析。
在类的准备阶段,会给Parent的静态变量p,n,m分配内存,并赋初始值,p赋值为null,n,m赋值为0;
程序执行Parent.getInstance()的时候是对类的主动使用,将导致类的初始化,Parent类中按照程序员的意愿给静态变量赋初值,调用Parent的构造方法,将n,m各自加1,这是n=1,m=1。然后按照顺序执行static int n;这时n的值并没有改变,还是1;然后执行static int m=n+5;这时m的值被被修改,变为6。所有程序的输出为n=1,m=6。
相关文章推荐
- JVM虚拟机学习笔记01:类的加载、连接、初始化
- jvm学习之变量初始化(待补充)
- 学习笔记之线程、Thread类和线程终止相关整理(下)——线程异常&JVM停止
- JVM学习笔记之对象访问的两种方式
- C#学习之路,学习笔记 2.2 字符、字符串、数字的初始化与读取
- JVM学习笔记之二 -- 垃圾回收器
- 【Ubuntu+OpenCV】OpenCV之矩阵创建、初始化--学习笔记【4】
- java编程思想学习笔记(5)--初始化与清理
- coco2dx-2.2.2 win32启动过程(opengl 和 窗口大小初始化部分) - 学习笔记 1
- Effective C++学习笔记:尽量使用初始化而不要在构造函数里赋值
- java之jvm学习笔记七(jar包的代码认证和签名)
- uCOSIII学习笔记之任务堆栈的初始化
- spring源码学习笔记-初始化(四)-PostProcessor
- JVM垃圾收集算法学习笔记
- JVM学习笔记-内存溢出
- Java学习笔记(成员变量初始化补充)
- swift语言的学习笔记十三(初始化方法)
- Effective C# 学习笔记(十二) 多用成员变量初始化,少用指定赋值
- spring源码学习笔记-初始化(六)-完成及异常处理
- JVM学习笔记(三)------内存管理和垃圾回收