黑马程序员技术交流社区

标题: 静态代码块的执行顺序 [打印本页]

作者: 陈郊    时间: 2012-9-29 19:42
标题: 静态代码块的执行顺序
一直以来我我认为这三者的执行顺序是静态代码块、非静态代码块、构造函数。
网上搜了下也都是这么说的。
就像下面这段代码:
Java代码  
1.public class ExA {   
2.    static {   
3.        System.out.println("父类--静态代码块");   
4.    }   
5.  
6.    public ExA() {   
7.        System.out.println("父类--构造函数");   
8.    }   
9.  
10.    {   
11.        System.out.println("父类--非静态代码块");   
12.    }   
13.  
14.    public static void main(String[] args) {   
15.        new ExB();   
16.    }   
17.}   
18.  
19.class ExB extends ExA {   
20.    static {   
21.        System.out.println("子类--静态代码块");   
22.    }   
23.    {   
24.        System.out.println("子类--非静态代码块");   
25.    }   
26.  
27.    public ExB() {   
28.        System.out.println("子类--构造函数");   
29.    }   
30.}  
public class ExA {
static {
  System.out.println("父类--静态代码块");
}
public ExA() {
  System.out.println("父类--构造函数");
}
{
  System.out.println("父类--非静态代码块");
}
public static void main(String[] args) {
  new ExB();
}
}
执行结果
=====
父类--静态代码块
子类--静态代码块
父类--非静态代码块
父类--构造函数
子类--非静态代码块
子类--构造函数
可是静态代码块真的会一定在非静态代码块之前执行吗?
下面这段代码
Java代码  
1.public class ExA {   
2.    private static ExA a = new ExA();   
3.    static {   
4.        System.out.println("父类--静态代码块");   
5.    }   
6.  
7.    public ExA() {   
8.        System.out.println("父类--构造函数");   
9.    }   
10.  
11.    {   
12.        System.out.println("父类--非静态代码块");   
13.    }   
14.  
15.    public static void main(String[] args) {   
16.        new ExB();   
17.    }   
18.}   
19.  
20.class ExB extends ExA {   
21.    private static ExB b = new ExB();   
22.    static {   
23.        System.out.println("子类--静态代码块");   
24.    }   
25.    {   
26.        System.out.println("子类--非静态代码块");   
27.    }   
28.  
29.    public ExB() {   
30.        System.out.println("子类--构造函数");   
31.    }   
32.}  
public class ExA {
private static ExA a = new ExA();
static {
  System.out.println("父类--静态代码块");
}
public ExA() {
  System.out.println("父类--构造函数");
}
{
  System.out.println("父类--非静态代码块");
}
public static void main(String[] args) {
  new ExB();
}
}
class ExB extends ExA {
private static ExB b = new ExB();
static {
  System.out.println("子类--静态代码块");
}
{
  System.out.println("子类--非静态代码块");
}
public ExB() {
  System.out.println("子类--构造函数");
}
}
执行结果
=====
父类--非静态代码块
父类--构造函数
父类--静态代码块
父类--非静态代码块
父类--构造函数
子类--非静态代码块
子类--构造函数
子类--静态代码块
父类--非静态代码块
父类--构造函数
子类--非静态代码块
子类--构造函数
=====
可以发现非静态代码块并不是一定在静态代码块之后执行的。
我认为此时private static ExA a = new ExA()是静态变量。而java里面静态变量与静态代码块是按代码先后顺序执行
。所以就导致非静态代码块在静态代码块之前执行。


作者: 尤洋    时间: 2012-9-29 20:36
其实说白了 就是先加载类 后加载对象, 而静态部分随着类的加载而加载 优先于主函数
你代码中  把本类对象放到了类的成员位置,作为一个成员变量而存在
使其随着类一起加载,但是对象在类加载的时候尚不存在,为了向前拓宽 该对象的出现时间,所以给他加了一个static。
导致 该对象在加载类的过程中就被一起加载 并初始化了,这个时候的初始化调用了 非静态代码块 和构造函数。
这才出现了非静态出现在静态之前
本质原因还是  对象被作为类的成员随类一起加载了,而静态紧随其后被加载
作者: 崔朋朋    时间: 2012-9-29 23:27
本帖最后由 崔朋朋 于 2012-9-29 23:37 编辑

先加载类字节码,静态部分(静态代码快/静态成员变量)随着类的加载而加载,最后在去执行主函数中的代码(即创建对象引用,初始化成员变量等)。在静态部分是顺序执行的,即按照类文件中的代码编写先后顺序执行。

在你的第二段代码中是先有个静态成员变量并初始化了,对象在初始化的过程中会牵扯到非静态代码块和构造函数的执行,接下来才是静态代码快,在接下来就是主函数的执行了。所以会造成先执行非静态代码块,再执行静态代码块的 “假象”。

其实质还是先执行静态部分,再执行创建对象,初始化成员变量等再非静态部分,只是静态成员变量的初始化牵扯到了非静态部分的执行。

类加载执行顺序:类字节码→静态代码块static{...}/静态成员变量→创建对象引用,初始化成员变量






欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2