对象实例何时被创建,这个问题也许你用一句话就能回答完了。但是它的潜在陷阱却常常被人忽视,这个问题也许并不像你想的那么简单,不信请你耐心看下去。
我前几天问一个同学,是不是在调用构造函数后,对象才被实例化?他不假思索的回答说是。
请看下面代码:
Date date=new Date();
System.out.println(date.getTime());
新手在刚接触构造函数这个概念的时候。他们常常得出这样的结论:对象实例是在调用构造函数后创建的。因为调用构造函数后,调用引用(date)的实例方法便不会报NullPointerException的错误了。
二、经验者的观点
然而,稍稍有经验的Java程序员便会发现上面的解释并不正确。这点从构造函数中我们可以调用this关键字可以看出。
请看下面代码:
public class Test
{
public Test()
{
this.DoSomething();
}
private void DoSomething()
{
System.out.println("do init");
}
}
这段代码中我们在构造函数中已经可以操作对象实例。这也就证明了构造函数其实只是用于初始化,早在进入构造函数之前。对象实例便已经被创建了。
三、父类构造函数
当创建一个有父类的子类的时候。对象的实例又是何时被创建的呢?我们也许接触过下面经典的代码:
public class BaseClass
{
public BaseClass()
{
System.out.println("create base");
}
}
public class SubClass
{
public SubClass()
{
System.out.println("create sub");
}
public static void main(String[] args)
{
new SubClass();
}
}
结果是先输出create base,后输出create sub。这个结果看起来和现实世界完全一致,先有老爸,再有儿子。因此我相信有很多程序员跟我一样会认为new SubClass()的过程是:实例化BaseClass->调用BaseClass构造函数初始化->实例化SubClass->调用SubClass构造函数初始化。然而非常不幸的是,这是个错误的观点。
四、奇怪的代码
以下代码是为了驳斥上面提到的错误观点。但是这种代码其实在工作中甚少出现。
public class BaseClass
{
public BaseClass()
{
System.out.println("create base");
init();
}
protected void init() {
System.out.println("do init");
}
}
//
public class SubClass
{
public SubClass()
{
System.out.println("create sub");
}
@Override
protected void init()
{
assert this!=null;
System.out.println("now the working class is:"+this.getClass().getSimpleName());
System.out.println("in SubClass");
}
public static void main(String[] args)
{
new SubClass();
}
}
这段代码运行的结果是先调用父类的构造函数,再调用子类的init()方法,再调用子类的构造函数。
这是一段奇妙的代码,子类的构造函数居然不是子类第一个被执行的方法。我们早已习惯于通过super方便的调用父类的方法,但是好像从没这样尝试从父类调用子类的方法。
再次声明,这只是个示例。是为了与您一起探讨对象实例化的秘密。通过这个示例,我们再次印证了开头的观点:早在构造函数被调用之前,实例便已被创造。若该对象有父类,则早在父类的构造函数被调用之前,实例也已被创造。这让java显得有些不面向对象,原来老子儿子其实是一块儿出生的。
五、奇怪但危险的代码
本篇是对上篇奇怪代码的延续。但是这段代码更加具有疑惑性,理解不当将会让你出现致命失误。
请看下面代码:
public class BaseClass {
public BaseClass()
{
System.out.println("create base");
init();
}
protected void init() {
System.out.println("in base init");
}
}
public class SubClass extends BaseClass{
int i=1024;
String s="13 leaf";
public SubClass()
{
System.out.println("create sub");
init();
}
@Override
protected void init() {
assert this!=null;
System.out.println("now the working class is:"+this.getClass().getSimpleName());
System.out.println("in SubClass");
/////////////great line/////////////////
System.out.println(i);
System.out.println(s);
}
public static void main(String[] args) {
new SubClass();
//oh!my god!!
}
}
这段代码相比上一篇,只是在子类中添加了一些成员变量。而我们的目标正是集中在讨论成员变量初始化的问题上。
这段代码的执行顺序是:父类、子类实例化->调用父类构造函数->调用子类init()方法->调用子类构造函数->调用子类init()方法。最终的输出结果向我们揭示了成员变量初始化的秘密。
当父类构造函数调用子类的init()方法的时候。子类的成员变量统统是空的,这个空是指的低级初始化。(值类型为0,布尔类型为false,引用类型为null)。而当子类构造函数调用init()方法的时候,成员变量才真正被初始化。这是一个危险的讯息,那就是使用父类构造函数调用子类时存在成员变量未初始化的风险。
我们的讨论也到此为止了。再次回顾,总结一下实例何时被创建这个问题。我得出了以下结论:
本文到此便结束了。鉴于本人才疏学浅,若是专业术语有错误,或是哪里讲的不对,也欢迎各位高手拍砖。
附上第五篇中SubClass的部分字节码,方便大家深入理解:
public SubClass();
0 aload_0 [this] //aload_0是啥?
1 invokespecial ques.BaseClass() [26] //调用父类构造函数
4 aload_0 [this]
5 sipush 1024 //初始化i成员变量
8 putfield ques.SubClass.i : int [28]
11 aload_0 [this]
12 ldc <String "13 leaf"> [30] //初始化s成员变量
14 putfield ques.SubClass.s : java.lang.String [32]
17 getstatic java.lang.System.out : java.io.PrintStream [34]
20 ldc <String "create sub"> [40]
22 invokevirtual java.io.PrintStream.println(java.lang.String) : void [42]
25 aload_0 [this]
26 invokevirtual ques.SubClass.init() : void [48] //调用init
29 return
分享到:
相关推荐
“反射”其实就是利用程序集的元数据信息。 反射可以有很多方法,编写程序时请先导入 System.Reflection 命名... // 创建类的实例 2、若要反射当前项目中的类(即当前项目已经引用它了)可以为: Assembly assembly
“System.NullReferenceException: 未将对象引用设置到对象的实例”问题可能原因如下: 1、ViewState 对象为Null。 2、DateSet 空。 3、sql语句或Datebase的原因导致DataReader空。 4、声明字符串变量时未赋空值就...
* 执行启动类的main函数 -> 创建对象的继承树从高到底的类层次中的静态块(如果已经被执行过,则不会重复执行) -> * 继承树中的从最高层开始的实例语句块{}、然后对应的构造函数、以及构造函数中调用的方法 * ...
对oracle数据库的操作经常会遇到trigger ,procude 等对象的创建,该文档是很有帮助的。
解决打开U8模块后提示“不能创建对象实例”,重新注册dll的批处理脚本。
qt通过类名动态创建对象
PHP把实例化的class对象存入session(session的值写入memcache),不必每次都创建类的实例 听起来有点绕口,其实就是: 1.不管你网站有多大,且有多少个class,用此方法之后每个类只需要实例化一次(除非你重启了服务器),听...
Dreamweaver 8中文版实例教程 第3章 创建网页基本对象
//a是Null对象 protected void Page_Load(object sender, EventArgs e) { a.ToString();//调用一个Null对象的方法 } 当然啦!结果就如下图了: 这么赤裸裸的写出这种代码,不太容易,通常更倾向于下面一种: 2:...
C++大学课本中非常实用又非常难理解的动态创建对象,我做了个简单的实例供大家参考学习
如下所示: 代码如下: var person...上面的例子创建了一个名为person的对象,并为它添加了三个属性(name、age和job)和一个方法(sayName())。其中,sayName()方法用于显示this.name()的值。早期的JavaScript开发人员
创建三维图形对象.xlsx源码EXCEL VBA宏编程xlsx实例代码下载创建三维图形对象.xlsx源码EXCEL VBA宏编程xlsx实例代码下载创建三维图形对象.xlsx源码EXCEL VBA宏编程xlsx实例代码下载创建三维图形对象.xlsx源码EXCEL ...
大家都知道抽象类无法实例化,就无法创建对象。所以下面这篇文章主要给大家介绍了关于Java实例化一个抽象类对象的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面随着小编来一起学习学习吧...
EXCEL 对象操作实例 C# VB 详细介绍excel对象 及使用实例 并介绍如何创建。NET 的 excel ADD_IN
python3 对象与实例 Python3是一种面向对象的编程语言,它的核心思想是将现实世界中的事物抽象成对象,通过对象之间的交互来完成程序的功能。在Python3中,对象和实例是面向对象编程中最基本的概念之一,本文将详细...
主要介绍了Java创建内部类对象实例详解的相关资料,需要的朋友可以参考下
C#,利用反射动态创建对象 C#,利用反射动态创建对象 C#,利用反射动态创建对象 C#,利用反射动态创建对象
创建对象 字段赋值 属性赋值 方法调用 事件订阅 事件注销
Spring IOC 04 配置通过静态工厂方法创建的bean、实例工厂创建的bean、FactoryBean ...(4)实例工厂:工厂本身需要创建对象;如下: 工厂类 工厂对象 = new 工厂类(); 工厂对象.getAirPlane(“张三”);
在Main类的main()方法中,我们创建了两个Person类的实例对象,分别为person1和person2。我们设置了对象的属性值,并通过调用对象的方法展示了对象的行为这个示例展示了如何使用Java中的类和对象。类定义了对象的属性...