Java的声明和初始化:细看OO程序执行的顺序

开发 后端
本文通过一个Base和Derived类的实例,介绍了在Java的声明和初始化过程中的程序执行顺序。在面向对象的世界中,程序执行的顺序相当的重要。

在介绍Java的声明和初始化的执行顺序之前,让我们先来看两个类:Base和Derived类。注意其中的whenAmISet成员变量,和方法preProcess()

public class Base  
{  
    Base() {  
        preProcess();  
    }  
 
    void preProcess() {}  

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
public class Derived extends Base  
{  
   public String whenAmISet = "set when declared";  
 
   @Override void preProcess()  
   {  
       whenAmISet = "set in preProcess()";  
   }  

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.

如果我们构造一个子类实例,那么,whenAmISet 的值会是什么呢?

public class Main  
{  
   public static void main(String[] args)  
   {  
       Derived d = new Derived();  
       System.out.println( d.whenAmISet );  
   }  

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.

再续继往下阅读之前,请先给自己一些时间想一下上面的这段程序的输出是什么?是的,这看起来的确相当简单,甚至不需要编译和运行上面的代码,我们也应该知道其答案,那么,你觉得你知道答案吗?你确定你的答案正确吗?

很多人都会觉得那段程序的输出应该是“set in preProcess()”,这是因为当子类Derived 的构造函数被调用时,其会隐晦地调用其基类Base的构造函数(通过super()函数),于是基类Base的构造函数会调用preProcess() 函数,因为这个类的实例是Derived的,而且在子类Derived中对这个函数使用了override关键字,所以,实际上调用到的是:Derived.preProcess(),而这个方法设置了whenAmISet 成员变量的值为:“set in preProcess()”。

当然,上面的结论是错误的。如果你编译并运行这个程序,你会发现,程序实际输出的是“set when declared ”。怎么为这样呢?难道是基类Base 的preProcess() 方法被调用啦?也不是!你可以在基类的preProcess中输出点什么看看,你会发现程序运行时,Base.preProcess()并没有被调用到(不然这对于Java所有的应用程序将会是一个***灾难性的Bug)。

虽然上面的结论是错误的,但推导过程是合理的,只是不完整,下面是整个运行的流程:

◆进入Derived 构造函数。

◆Derived 成员变量的内存被分配。

◆Base 构造函数被隐含调用。

◆Base 构造函数调用preProcess()。

◆Derived 的preProcess 设置whenAmISet 值为 “set in preProcess()”。

◆Derived 的成员变量初始化被调用。

◆执行Derived 构造函数体。

等一等,这怎么可能?在第6步,Derived 成员的初始化居然在 preProcess() 调用之后?是的,正是这样,我们不能让成员变量的声明和初始化变成一个原子操作,虽然在Java中我们可以把其写在一起,让其看上去像是声明和初始化一体。但这只是假象,我们的错误就在在我们把Java的声明和初始化看成了一体。在C++的世界中,C++并不支持成员变量在声明的时候进行初始化,其需要你在构造函数中显式的初始化其成员变量的值,看起来很土,但其实C++用心良苦。

在面向对象的世界中,因为程序以对象的形式出现,导致了我们对程序执行的顺序雾里看花。所以,在面向对象的世界中,程序执行的顺序相当的重要。

下面是对上面各个步骤的逐条解释。

◆进入构造函数。

◆为成员变量分配内存。

◆除非你显式地调用super(),否则Java 会在子类的构造函数最前面偷偷地插入super() 。

◆调用父类构造函数。

◆调用preProcess,因为被子类override,所以调用的是子类的。

◆于是,初始化发生在了preProcess()之后。这是因为,Java需要保证父类的初始化早于子类的成员初始化,否则,在子类中使用父类的成员变量就会出现问题。

◆正式执行子类的构造函数(当然这是一个空函数,居然我们没有声明)。

你可以查看《Java语言的规格说明书》中的 相关章节 来了解更多的Java创建对象时的细节。

***,需要向大家推荐一本书,Joshua Bloch 和 Neal Gafter 写的 Java Puzzlers: Traps, Pitfalls, and Corner Cases,中文版《JAVA解惑》。

【编辑推荐】

  1. Java程序员面试必备的32个要点
  2. 可能不再有Java SE 7?甲骨文面临Java许可问题
  3. Java未来的三大谜题:再谈甲骨文收购Sun
  4. 浅谈Java线程的生命周期
  5. 关于Java继承的一些复习
责任编辑:yangsai 来源: 酷壳
相关推荐

2012-02-28 10:04:09

Java

2009-06-11 13:26:16

Java数组声明创建

2013-03-04 11:10:03

JavaJVM

2010-07-28 10:22:33

FlexApplica

2009-10-20 14:03:48

VB.NET数组声明VB.NET数组初始化

2011-03-23 15:02:55

ListenerFilterServlet

2012-03-13 13:38:42

Java

2022-01-04 19:33:03

Java构造器调用

2011-06-09 14:13:06

C++JAVA缺省初始化

2021-04-07 08:03:51

js举起Hoisting初始化

2011-07-22 17:46:43

java

2012-05-23 12:46:53

JavaJava类

2015-08-14 14:31:57

Java初始化面试题

2015-10-30 09:51:19

Java重写初始化隐患

2019-11-04 13:50:36

Java数组编程语言

2009-08-26 18:28:44

C#数组

2011-06-17 15:29:44

C#对象初始化器集合初始化器

2021-07-07 05:00:17

初始化源码

2009-06-10 16:17:00

Netbeans JT初始化

2011-03-16 10:52:20

点赞
收藏

51CTO技术栈公众号