本文主要介绍以下两块内容的执行顺序,熟悉的大虾可以直接飘过。
一。JAVA中执行顺序
静态块
块
构造器
父类构造器
二。JAVA中赋值顺序
静态块直接赋值
块直接赋值
父类继承的属性已赋值
静态变量声明时赋值
成员变量声明时赋值
构造器赋值
第一部分很好测试,我们只需要写一个子类,类中定义一个静态块,一个普通块,一个构造器,它的父类构造器,都打印一条语句,即可明白它们直接的执行顺序
Mastiff类
Java代码 收藏代码
<span style=”font-size: medium;”>/**
* 子类藏獒
*/
public class Mastiff extends Dog {
public Mastiff() {
System.out.println(“Mastiff”);
}
{
System.out.println(“block”);
}
static {
System.out.println(“static block”);
}
public static void main(String[] args){
Mastiff mastiff=new Mastiff();
}
}
</span>
DOG类
Java代码 收藏代码
<span style=”font-size: medium;”>/**
*DOG父类
*/
public class Dog {
public Dog() {
System.out.println(“Dog”);
}
}
</span>
运行结果为:
static block
Dog
block
Mastiff
也就是说,在我们的程序中,实例化一个类对象的时候,运行顺序为:
静态块
父类构造器
本类中的块
本类的构造器
我们可以更进一步,如果在父类中也有块和静态块呢?
DOG类改进后源码
Java代码 收藏代码
<span style=”font-size: medium;”>/**
*DOG父类
*/
public class Dog {
public Dog() {
System.out.println(“Dog”);
}
static{
System.out.println(“super static block”);
}
{
System.out.println(“super block”);
}
}
</span>
Mastiff改进后源码
Java代码 收藏代码
<span style=”font-size: medium;”>/**
* 子类藏獒
*/
public class Mastiff extends Dog {
public Mastiff() {
System.out.println(“Mastiff”);
}
{
System.out.println(“block”);
}
static {
System.out.println(“static block”);
}
public static void main(String[] args){
Mastiff mastiff=new Mastiff();
}
}
</span>
运行的结果为:
super static block
static block
super block
Dog
block
Mastiff
也就是说此时的运行顺序为:
父类静态块
自身静态块
父类块
父类构造器
自身块
自身构造器
好了,知道了运行的顺序,那么这是为什么呢?
这就要从JVM中类的装载机制和实例化机制开始说起,这里因为主题原因,先不讨论,有兴趣的同学可以自己查资料。
我们再来讨论第二个问题,一个变量的值,它有可能在哪些地方确定呢??
从父类继承该值(包括:1.作为父类的成员变量已经赋值 2.在父类的块中赋值 3.在父类的构造器中赋值)
在构造器中对其进行赋值
在块中进行赋值
在方法调用中进行赋值
现在假设在我们刚刚的例子中,有一个变量type,表示狗的品种
Java代码 收藏代码
<span style=”font-size: medium;”>/**
*DOG父类
*/
public class Dog {
public String type=”父类成员变量赋的值”;
public Dog() {
System.out.println(“父类构造器–type–>”+type);
type=”父类构造器赋的值”;
System.out.println(“父类构造器—-type—>”+type);
}
{
System.out.println(“block—type—>”+type);
type=”父类块赋的值”;
}
}
</span>
Java代码 收藏代码
<span style=”font-size: medium;”>/**
* 子类藏獒
*/
public class Mastiff extends Dog {
public String type=”成员变量赋的值”;
public Mastiff() {
System.out.println(“构造器—type—>”+type);
type=”构造器赋的值”;
}
public void say(){
System.out.println(“say—type—->”+type);
}
{
System.out.println(“block—type—>”+type);
type=”块赋的值”;
}
public static void main(String[] args){
Mastiff mastiff=new Mastiff();
mastiff.say()</span><span style=”font-size: medium;”>;</span><span style=”font-size: medium;”>
}
}
</span>
执行结果如下:
block—type—>父类成员变量赋的值
父类构造器–type–>父类块赋的值
父类构造器—-type—>父类构造器赋的值
block—type—>成员变量赋的值
构造器—type—>块赋的值
say—type—->构造器赋的值
答案很明显,赋值顺序为:
父类成员变量赋值
父类块赋值
父类构造器赋值
自身成员变量赋值
自身块赋值
自身构造器赋值
结合我们前面说的程序中的执行顺序,这个显然是很好理解的:
1.成员变量赋值>>>块赋值>>>构造器赋值
2.父类的块>>父类构造器>>自身块>>自身构造器
又因为一个成员变量是不可能在静态变量中赋值的,而且又前面程序执行顺序可知
静态块>>块
所以,程序的赋值步骤为
父类的静态变量赋值
自身的静态变量赋值
父类成员变量赋值
父类块赋值
父类构造器赋值
自身成员变量赋值
自身块赋值
自身构造器赋值