Java子类和父类的初始化执行顺序
要明白子类和父类的初始化执行顺序,只需要知晓以下三点,就不会再弄错了。
1.创建子类对象时,子类和父类的静态块和构造方法的执行顺序为:父类静态块->子类静态块->父类构造器->子类构造器。深入理解为什么是这个顺序,可以看我这篇文章:从京东面试题看java类和对象的初始化
2.静态变量的声明和赋值,声明会在静态块之前,赋值运算将会合并到静态块中,顺序和源代码中的顺序一致。举例如下:
源代码
01
02
03
04
05
06
07
08
09
10
|
public class P { public static int a = 1 ; static { int b = 2 ; } public static int c = 3 ; } |
在编译器编译后,会变成这样子
01
02
03
04
05
06
07
08
09
10
11
12
|
public class P { public static int a; public static int c; static { a = 1 ; int b = 2 ; c = 3 ; } } |
我们来看,编译后的字节码是怎样的,使用命令可以反编译类的字节码:javap -v -p P.class
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
|
{ public static int a; descriptor: I flags: ACC_PUBLIC, ACC_STATIC public static int c; descriptor: I flags: ACC_PUBLIC, ACC_STATIC static {}; descriptor: ()V flags: ACC_STATIC Code: stack= 1 , locals= 1 , args_size= 0 0 : iconst_1 1 : putstatic # 2 // Field a:I 4 : iconst_2 5 : istore_0 6 : iconst_3 7 : putstatic # 3 // Field c:I 10 : return |
我去掉了编译器生成的构造方法以及一些无关信息,我们可以看到字节码中,a、c的声明在前面(其实这个不是重点),在static{}块中,pc 0~1两个指令,为静态字段a赋值1,pc 4~5两个指令,为第一个局部变量,也就是变量b赋值2,pc 6~7两个指令,为静态字段c赋值3。可以看到合并后的static块,a的赋值在原静态块代码之前,c的赋值在原静态块代码之后,这个顺序和源代码中ac的声明顺序一致。
3.成员变量的声明和赋值,与静态变量相同的是成员变量的赋值也会合并到构造器中,不同的是合并后的顺序,成员变量的赋值是在构造器的前面。举例如下:
源代码
1
2
3
4
5
6
7
8
9
|
public class P { public int a = 1 ; public P() { int b = 2 ; } public int c = 3 ; |
编译后的代码,会像这个样子
01
02
03
04
05
06
07
08
09
10
11
|
public class P { public int a; public int c; public P ( ) { a = 1 ; c = 3 ; int b = 2 ; } } |
再来看看编译后的字节码是怎样的
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
public int a; descriptor : I flags : ACC_PUBLIC public int c; descriptor : I flags : ACC_PUBLIC public P ( ) ; descriptor : ( ) V flags : ACC_PUBLIC Code : stack = 2 , locals = 2 , args_size = 1 0 : aload_ 0 1 : invokespecial #1 // Method java/lang/Object."<init>":()V 4 : aload_ 0 5 : iconst_ 1 6 : putfield #2 // Field a:I 9 : aload_ 0 10 : iconst_ 3 11 : putfield #3 // Field c:I 14 : iconst_ 2 15 : istore_ 1 16 : return |
字段a和c的声明在前面,然后看构造器P()的字节码,pc 0~1两个指令,是先调用P的父类Object的构造器,字节码中的构造器方法使用<init>来表示的。pc 4~6三个指令,为成员变量a赋值1。pc9~11三个指令,为成员变量c赋值3,pc 14~15两个指令,为下表为1的局部变量赋值为2,也就是局部变量b=2。所以可以看出,成员变量赋值逻辑合并到构造器后,是在调用父类构造器之后,原有构造器代码之前。
回过头来,你明白了子类父类初始化各个方法的执行顺序,而字段的初始化赋值也是合并到方法里,所以创建子类对象时,子类父类各个部分的执行顺序都已了然。
总结:
1.讲解了子类父类初始化时方法执行顺序,包括的静态块和构造器方法,静态块也是方法,静态块在jvm中的方法名叫<cinit>。
2.讲解了字段的赋值是如何合并到方法中,静态字段赋值合并到静态块中,成员变量赋值合并到构造器方法中。
更多java学习资料可关注:itheimaGZ获取(公冢号)
Java子类和父类的初始化执行顺序的更多相关文章
- Java中子类和父类相关方法的执行顺序
无意中看到下面一个题目,大家一起来看看最后的输出结果是什么.反正我看完之后,用IDE测试后感觉知识点得到巩固了. /** * 函数执行顺序测试 * Created by 萌小Q on 2017/5/1 ...
- Java的初始化执行顺序(父类static变量->子类static变量->父类成员变量->父类构造器->成员变量->构造器->main函数)
1. 引言 了解Java初始化的顺序,有助于理解Java的初始化机制和内存机制. 顺序:父类static变量->子类static变量->父类成员变量->父类构造器->成员变量- ...
- 关于java中构造方法、实例初始化、静态初始化执行顺序
在Java笔试中,构造方法.实例初始化.静态初始化执行顺序,是一个经常被考察的知识点. 像下面的这道题(刚刚刷题做到,虽然做对了,但是还是想整理一下) 运行下面的代码,输出的结果是... class ...
- Java拾遗(一):浅析Java子类和父类的实例化顺序 及 陷阱
本文主要介绍Java里经常使用的子类和父类的变量实例化顺序及陷阱,并结合一个Android实例来探讨此问题.日后编程中应尽量避免此陷阱. 首先看以下一段代码: 定义一个虚类Server.java pa ...
- java 子类、父类中静态代码块、字段,非静态代码块、字段以及构造函数的初始化顺序和次数
一个类中的数据初始化顺序是面试官非常喜欢出的面试题之一,本文用一个实例来介绍java中子类.父类中静态代码块.字段,非静态代码块.字段以及构造函数的执行顺序和次数. 一.包结构
- Java 中变量初始化、子类和父类构造器调用的顺序
先说结论 变量初始化 -> 父类构造器 -> 子类构造器 贴代码 Animcal.java 父类 public class Animal { private static int inde ...
- java子类和父类中静态块、非静态块、构造函数的执行顺序
public class qqqq extends Parent{ public static void main(String[] args) { new Child(); } } class Pa ...
- Java子类继承父类的执行顺序
父类的静态代码块(static) 子类的静态代码块(static) 父类的非静态代码块(父类成员初始化) 父类的构造方法 子类的非静态代码块(子类成员初始化) 子类的构造方法
- odoo开发笔记 -- 多个子类继承同一个父类方法的执行顺序
场景描述: odoo模块化开发的架构理念,科学&高效, 可以让很多业务场景,尽可能松耦合:让开发人员的主要精力,关注在当前的业务逻辑: 所谓「前人栽树,后人乘凉」,模块整体好比一棵大树, 开发 ...
随机推荐
- h5与安卓、ios交互
1.安卓交互 h5调用安卓方法 window.webview.xxx() 安卓调用h5方法, 方法需要在全局注册 window['showUnreadMsg'] = () => { this.$ ...
- Linux--计划任务未执行
参考:http://blog.csdn.net/shangdiyisi/article/details/9477521 日志 /var/log/cron
- Linux-让程序不能多次运行
1.因为守护进程是长时间运行而不退出的,因此./a.out执行一次就有一个进程,执行多次就有多个进程. 2.这样并不是我们想要的.我们的守护进程一般都是服务器,服务器程序只要运行一个就够了,多次同时运 ...
- mac下使用opencv编译安装新模块contrib
opencv-4.0.1 opencv_contrib-4.0.1 提供ippicv下载链接: https://pan.baidu.com/s/1OIJRUqPqAtpMetku8qX36w cont ...
- 函数返回值return
#函数后面如果没有return系统会默认return none def ff(): print("打印return") return 15 # 函数在执行中遇到return就会停止 ...
- linux 查看链接库的版本
我们编译可执行文件的时候,会链接各种依赖库, 但是怎么知道依赖库的版本正确呢? 下面有几种办法: ldd 这是比较差的,因为打印结果更与位置相关 dpkg -l | grep libprotobuf ...
- PowerShell创建 Profile
profile主要用于个性化常用的函数.别名等等.每次加载powershell的时候,都会执行profile中的内容. 查看是否有profile: $profile 如果结果是false说明没有.则创 ...
- Python学习中的随笔,好记性不如烂笔头
本文 为博主看了 vamei 的blog 写下的随笔 . 致敬Vamei 1.type() 可以显示参数的类型 如 : a=12 type(a) 为 int 2.python的基本类型 为 i ...
- JavaScript学习笔记 - 进阶篇(6)- JavaScript内置对象
什么是对象 JavaScript 中的所有事物都是对象,如:字符串.数值.数组.函数等,每个对象带有属性和方法. 对象的属性:反映该对象某些特定的性质的,如:字符串的长度.图像的长宽等: 对象的方法: ...
- upstream(负载均衡)
一.什么是负载均衡 负载均衡,顾名思义是指将负载尽量均衡的分摊到多个不同的服务器,以保证服务的可用性和可靠性,提供给客户更好的用户体验: 负载均衡的直接目标就是尽量发挥多个服务单元的整体效能,要实现这 ...