java类中包含静态成员、方法时的打印顺序

来源:互联网 发布:淘宝网买运动鞋 编辑:程序博客网 时间:2024/06/06 07:20
class Base{    static{         System.out.println("{父类加载}");         }     static int a=getValue("静态:获取父类a值");     Base(){          System.out.println("调用父类无参构造函数,现打印父类a赋值结果:"+a);          a++;          printA();      }     {          System.out.println("{父类实例初始代码块}");          }     Base(String s){          System.out.println("父类带参构造函数"+s);          }     void printA(){         System.out.println("父类赋值结果a="+a);     }     static int getValue(String s){          System.out.println("静态:父类原始a="+a);          System.out.println(s);          return ++a;      } }public class Child extends Base{    static{         System.out.println("{子类加载}");         }     static Child child=new Child();     static int a=getValue("静态:获取子类a值");         Child(){         System.out.println("调用子类无参构造函数,现打印子类a赋值结果:"+a);         a++;         printA();     }    {          System.out.println("{子类实例初始代码块}");          }     Child(String s){          System.out.println("子类带参构造函数"+s);          }    void printA(){         System.out.println("子类赋值结果a="+a);     }    static int getValue(String s){         System.out.println("静态:子类原始a="+a);         System.out.println(s);         return ++a;     }    public static void main(String[] args){         //new Child("实例化子类");         new Child();     } }


打印结果如下:

---------------------------------------------------------

{父类加载}
静态:父类原始a=0
静态:获取父类a值
{子类加载}
{父类实例初始代码块}
调用父类无参构造函数,现打印父类a赋值结果:1
子类赋值结果a=0
{子类实例初始代码块}
调用子类无参构造函数,现打印子类a赋值结果:0
子类赋值结果a=1
静态:子类原始a=1
静态:获取子类a值
{父类实例初始代码块}
调用父类无参构造函数,现打印父类a赋值结果:2
子类赋值结果a=2
{子类实例初始代码块}
调用子类无参构造函数,现打印子类a赋值结果:2
子类赋值结果a=3

-------------------------------------------------------------------

 

例子中有如下几处刻意设计的难点:

1.父子类都有同名变量a,考察静态变量在继承中的表现;

2.静态成员child在初始化时即被实例化,先于静态变量a的赋值,考察a的赋值顺序;

3.静态代码块和实例代码块的区别;

4.printA()是实例方法,考察当实例化子类时,在父类构造方法中调用实例方法是否有多态;

 

分析

1.继承顺序:当实例化子类时,会先加载父类,并给父类的静态变量分配空间和自动初始化,其值和实例变量初始化值相同,初始化时可以调用静态方法;然后才加载子类,并给子类的静态变量分配空间,此时如果静态的类成员需要实例化,则调用子类构造方法。而进入子类构造方法时,最先调用super(),所以进入父类无参构造方法,此时调用的实例方法一定是子类重写的方法,因为堆中只有子类的实例方法。

 

2.初始化顺序:当静态的类成员和静态的基本变量同时初始化时,其先后顺序取决于语句顺序。

 

3.实例代码块仅当构造方法中调用super()之后才被调用,每次实例化都会被调用一次;而静态代码块则只在类被第一次加载时调用,且只运行一次。

 

4.静态变量:静态变量属于类,存在于代码区,内存中只有1份,为所有实例共享,任何实例对其修改都会影响到它的值。

 

5.语句顺序:

static Child child=new Child();

static int a=getValue("静态:获取子类a值");

注意这里的顺序导致了子类加载之后立即在堆中实例化了一个child对象,而在代码区中并未执行到静态变量a的声明,即:

static int a=getValue("静态:获取子类a值");

故先后出现父子类构造函数的打印语句,然后才是"静态:获取子类a值"。