###############################

今天总结一下,new对象的初始化过程。

###############################

首先,当不含static成员时,先看一下案例图:

Java new 对象初始值 new对象初始化_初始化

如图所示,我们先定义相关类,Building,House(继承Building),Villa(继承House),同时,House有成员变量LivingRoom,LivingRoom有成员变量Bed和Desk。

具体代码实现如下:

1 //建筑
 2 class Building {
 3     public Building() {
 4         System.out.println("Building");
 5     }
 6 }
 7 //房子
 8 class House extends Building{
 9     public LivingRoom livingRoom = new LivingRoom();//卧室
10     public House() {
11         System.out.println("House");
12     }
13 } 
14 //别墅
15 class Villa extends House{
16     public Villa () {
17         System.out.println("Villa");
18     }
19 }
20 //卧室
21 class LivingRoom {
22     public Bed bedFirst = new Bed();//床
23     public Desk deskFirst = new Desk();//桌子
24     
25     public LivingRoom () {
26         System.out.println("LivingRoom");
27     }
28 }
29 //床
30 class Bed {
31     public Bed () {
32         System.out.println("Bed");
33     }
34 }
35 //桌子
36 class Desk {
37     public Desk () {
38         System.out.println("Desk");
39     }
40 }
41 此时,我们new Villa():
42 public class InitializeDemo {
43     @Test
44     public void testInitialize() {
45         Villa villa = new Villa();
46     }
47 }
48 执行结果为:
49 Building
50 Bed
51 Desk
52 LivingRoom
53 House
54 Villa

 

 在此案例中,我们在创建Villa对象时,会先试着创建Villa的父类House的对象,但House也有父类Building,Building还有父类object。因此,实际上,本类会先创建object对象并初始化,在object对象创建并初始化后,会创建building对象,building对象没有成员变量,所以直接执行构造器,打印出building。

  接着,开始创建house对象,house对象有成员变量livingRoom,因此,会先初始化livingRoom,创建livingRoom时,也会先创建livingRoom的父类object对象并初始化没其次再创建livingRoom实例,livingroom同样有两个成员变量bed和desk,因而会按两个成员变量的定义顺序,先后创建bed和desk。初始化成员变量成功后,执行livingroom的构造器,打印livingroom。初始化livingroom成功后,开始执行house的构造器,打印house。

  最后,在house初始化成功后,执行了Villa的构造器,打印出Villa。

  初始化成功。

其次,当有static成员时

1 class Family {
 2     Family (Class className) {
 3         System.out.println("创建Family对象");
 4         System.out.println(className.getName());
 5     }
 6 }
 7 class Person {
 8     Person() {
 9         System.out.println("创建Person对象");
10     }    
11     public static Family family = new Family(Person.class);
12     static {
13         System.out.println("执行Person静态代码块");
14     }
15     {
16         System.out.println("执行Person代码块");
17     }
18 } 
19 public class Student extends Person {    
20 public Student() {
21         System.out.println("创建student对象");
22     }
23     {
24         System.out.println("执行Student代码块");
25     }
26     static {
27         System.out.println("执行Student静态代码块");
28     }
29     public static Family staticFamily = new Family(Student.class);
30     @Test
31     public void test() {
32         System.out.println("=============");
33     }
34     public Family family = new Family(Student.class);
35 }
36 执行结果:
37 创建Family对象
38 com.tca.thinkInJava.chap7.Person
39 执行Person静态代码块
40 执行Student静态代码块
41 创建Family对象
42 com.tca.thinkInJava.chap7.Student
43 执行Person代码块
44 创建Person对象
45 执行Student代码块
46 创建Family对象
47 com.tca.thinkInJava.chap7.Student
48 创建student对象

 

1.首先在执行@test方法时,必须创建student的实例对象。

  2.要创建student的实例对象,第一步先加载student父类person和student类的字节码,并对相关静态成员变量进行初始化,执行静态代码块。

  3.首先加载父类person字节码,静态成员变量public static family ,family优于静态代码块前声明,所以先对静态成员变量family进行初始化。创建family对象,打印“创建family对象”,“xxxxxx”。

  4.加载完person字节码后,再加载student字节码。静态代码块再静态成员public static family staticfamily之前声明,所以先执行静态代码块,打印“执行student静态代码块”,在进行成员变量初始化,执行“创建family对象”,“xxxxxxxxxx”。

  5.创建student父类person的实例对象,创建实例对象时,先进行非静态成员变量初始化和代码块的执行,在执行构造器方法。所以先打印“执行person代码块”,再打印“创建person对象”。父类对象创建和初始化完毕。

  6.创建子类对象student,先进行非静态成员变量初始化和代码块的执行,在执行构造方法。代码块的声明再非静态成员变量声明之前,所以先执行代码块,打印“执行student代码块”,再对family进行初始化,创建family对象,打印“创建family对象”,“xxxxxxxxxxx”。

  7.执行test测试方法。

 

总结:

  当不含static成员时,我们new一个对象A时,首先会先创建A类的父类B的实例对象,如果B类仍有C类,会先创建父类C的对象,以此类推,是一个递归创建的过程;当该类的父类对象已经全部创建并初始化时,会对该类进行创建和初始化。但在对该类进行初始化时,会先初始该类对象的成员变量,再执行该类的构造方法。成员变量的初始化和代码块的执行顺序,由它们声明顺序决定的,按顺序依次初始化或执行,但均在构造器方法之前执行。

  当有static成员时,在该类第一次被JVM加载时,静态成员变量只会被初始化一次,静态代码块只会被执行一次。会先加载父类字节码,再加载子类字节码,如果有创建某类的实例对象,也是在该类的父类和该类的字节码加载完成之后,才会实例化该类父类的实例对象和该类的实例对象。静态成员变量的初始化和静态代码块的执行顺序,是由它们的声明顺序决定的,按顺序依次初始化或执行。