对象的创建过程

来源:互联网 发布:许昌永诚网络 编辑:程序博客网 时间:2024/06/08 00:43

        一个对象的创建语句如下:

Point originOne = new Point(23, 94);
       该语句实际上分三部分Declaration:声明、Instantiation:实例化、Initialization:初始化。

        ----Declaration: 声明一个变量。

        ----Instantiation:new运算符实例化一个对象,给该新对象分配内存空间并返回该内存空间的引用。注意这一步就完成了默认初始化。

        ----Initialization:new运算符后跟着的构造函数进行对象的初始化。

       题外话,这里可以理解到一个对象的创建不是一个原子语句,所以饿汉Singleton模式创建的时候(代码如下所示),在多线程环境下,即使对象已经不是null了仍然有可能失败,因为这个时候对象还未被初始化。解决这个问题其实可以用final关键字来解决。当变量声明final后,其他线程会在构造函数完成构造之后才能访问该单例变量。

package com.creation.singleton;import java.util.Scanner;public class Singleton {//注意此处的关键字finalprivate final static Singleton INSTANCE = new Singleton();private Singleton() {Scanner s = new Scanner(System.in);}public static synchronized Singleton getInstance() {return INSTANCE;}}

       同时在一个对象实例化(Instantiation)后,整个对象其实就已经完全被创建好了,其函数可以被调用,只是其内部的变量是系统默认初始化的值。所以在该篇文章中实例变量可以被内部函数的返回值来初始化。

针对刚才实例化的时候的默认初始化就已经完成的问题有一个面试题

public class Main {public static class Father {private int fatherVal = 1;public Father() {test();}public void test() {System.out.println(fatherVal);}}public static class Son extends Father {private int sonVal = 2;public Son() {test();}public void test() {System.out.println(sonVal);}}/** * @param args */public static void main(String[] args) {// TODO Auto-generated method stubnew Son();}}
问题是输出是什么?

分析:

第一步 实例化Son的空间,该实例化也包括了原来父类的部分。这里实例化的时候就已经把所有的变量赋值为默认初始值了;

第二步 先执行父类的显示赋值语句和构造函数,在父类的构造函数中调用test()函数,子类覆盖了父类的test()函数则调用的是Son的test()函数。关键点就是,此时Son部分还没有进行初始化(这里的初始化指的是,指定初始化,静态语句块、非静态语句块,构造函数),所以sonVal的值是0。所以输出是0 2。

注意:这里其实constructor这个名字翻译得不好,容易让人误解,对象是通过构造函数构建出来的一样,其实构造函数是初始化过程中被使用的。