最近在看《Thinking in Java》这本书,之前一直对类及对象的初始化过程不太清楚,只是感到很模糊。看了这本书关于对象初始化的部分,终于搞明白了。

废话不多说,先上两个例子,实例摘自 《Thinking in Java》。

Test1:

 1 package com.westward;
 2 /**
 3  * @desc 《Thinking in Java》 Demo Page 115
 4  * */
 5 public class Demo14 {
 6 
 7     public static void main(String[] args) {
 8         // TODO Auto-generated method stub
 9         System.out.println("Inside main");//先初始化static成员变量和static代码块,然后非static成员变量和代码块,然后构造器,然后static main方法
10         Cups.c1.f(99);//1
11     }
12     
13     /*static Cups x=  new Cups();
14     static Cups y=     new Cups();*///2
15 }
16 class Cup{
17     public Cup(int marker) {
18         System.out.println("Cup("+marker+")");
19     }
20     void f(int marker){
21         System.out.println("f("+marker+")");
22     }
23 }
24 class Cups{
25     static Cup c1;
26     static Cup c2;
27     static{
28         c1= new Cup(1);
29         c2= new Cup(2);
30     }
31     Cups() {
32         System.out.println("Cups()");// TODO Auto-generated constructor stub
33     }
34 }
35 //run as:Cup(1)  Cup(2)  Cups() Cups() Inside main f(99)
36 //注释掉tag2:Inside main Cup(1) Cup(2)  f(99) 

我们将Cups类定义为下面这个,其初始化过程是一样的。即先初始化static成员变量,然后才static代码块。

 1 class Cups{
 2     static{
 3         c1= new Cup(1);
 4         c2= new Cup(2);
 5     }
 6     static Cup c1;
 7     static Cup c2;
 8     Cups() {
 9         System.out.println("Cups()");// TODO Auto-generated constructor stub
10     }
11 }

 

Test2:

 1 package com.westward;
 2 
 3 /**
 4  * @desc 《Thinking in Java》 Demo Page 117
 5  * */
 6 public class Demo15 {
 7     Mug c1;
 8     Mug c2;
 9     {
10         c1= new Mug(1);
11         c2= new Mug(2);
12         System.out.println("c1 & c2 initialized");
13     }
14     Demo15(){
15         System.out.println("Mugs()");
16     }
17 
18     public static void main(String[] args) {
19         // TODO Auto-generated method stub
20         System.out.println("Inside main()");
21 //        Demo15 x= new Demo15();//2
22     }
23 
24 }
25 class Mug{
26     Mug(int marker) {
27         System.out.println("Mug("+marker+")");
28     }
29     void f(int marker){
30         System.out.println("f("+marker+")");
31     }
32 }
33 //Inside main() Mug(1) Mug(2) c1 & c2 initialized Mugs() 
34 //注释掉tag2: Inside main()

总结:从Test1和Test2,可以看出,首先,加载拥有主方法main的Demo14这个类,可以理解为jvm默认调用加载拥有main方法的Demo14这个类Demo14.class,然后初始化Demo14里的static修饰的成员变量,然后执行static修饰的代码块,然后执行main方法。main方法里(此处有三个出口):

  a.若定义一个类(比如Cups x),并new初始化这个对象Cups(),那么jvm就会加载Cups.class,然后初始化static成员变量,然后执行static修饰的代码块,然后非静态成员变量,然后非静态代码块,然后构造器。

  b.若直接调用类的静态方法或者静态类的静态成语,比如调用Cups.c1.f(99),那么jvm同样会加载Cups.class这个类,并且初始化。顺序为:初始化static成员变量,然后执行static修饰的代码块,然后非静态成员变量,然后非静态代码块,然后构造器。

  c.若main方法里既没有new对象,也没有调用类的静态成员或静态方法,没有类会被加载,所以没有类会被初始化,没有对象被初始化。

注意:

  1.上面的初始化过程,我们假定静态成员,静态代码块,非静态成员,非静态代码块都存在的情况。若某一项不存在,那么顺序是不会变的,我们只需要从这个顺序列表中将其删除就OK。

  2.当我们new同一个类(Cups)的多个对象,或者调用Cups的静态方法或者静态成员多次,Cups.class只加载并且初始化一次。也就是说static成员变量和static静态代码块只初始化一次。非静态的对象的非静态成员变量、非静态代码块、构造器初始化次数则和new的次数相等。

  3.当我们new了Cups的一个对象,jvm载入Cups.class,并且初始化static修饰的...后,然后会为非静态成员默认初始化值[基本数据类型(int,byte,short,long为0,float,double为0.0,boolean为false,char为''),引用数据类型会将句柄初始化为null],然后若我们显示初始化此成员变量,就会执行显示初始化,然后执行非静态代码块,然后构造器。

posted on 2016-03-13 18:16  WesTward  阅读(186)  评论(0编辑  收藏  举报