`

java类静态域、块,非静态域、块,构造函数的初始化顺序

    博客分类:
  • j2se
 
阅读更多

引用 :http://ini.iteye.com/blog/2007835

面试的时候,经常会遇到这样的考题:给你两个类的代码,它们之间是继承的关系,每个类里只有构造器方法和一些变量,

构造器里可能还有一段代码对变量值进行了某种运算,另外还有一些将变量值输出到控制台的代码,然后让我们判断输出的
结果。这实际上是在考查我们对于继承情况下类的初始化顺序的了解。 
我们大家都知道,对于静态变量、静态初始化块、变量、初始化块、构造器,它们的初始化顺序以此是 
(静态变量、静态初始化块)>(变量、初始化块)>构造器。 
我们也可以通过下面的测试代码来验证这一点:  
Java代码 

Java代码 复制代码 收藏代码
  1. public class InitialOrderTest {                
  2.     // 静态变量           
  3.     public static String staticField = "静态变量";           
  4.     // 变量           
  5.     public String field = "变量";                   
  6.     // 静态初始化块         
  7.     static {           
  8.         System.out.println(staticField);   
  9.         System.out.println("静态初始化块");   
  10.     }                   
  11.     // 初始化块               
  12.     {           
  13.         System.out.println(field);   
  14.         System.out.println("初始化块");         
  15.     }                   
  16.     // 构造器           
  17.     public InitialOrderTest() {   
  18.         System.out.println("构造器");      
  19.     }                   
  20.     public static void main(String[] args) {   
  21.         new InitialOrderTest();               
  22.     }           
  23. }  
public class InitialOrderTest {             
    // 静态变量        
    public static String staticField = "静态变量";        
    // 变量        
    public String field = "变量";                
    // 静态初始化块      
    static {        
        System.out.println(staticField);
        System.out.println("静态初始化块");
    }                
    // 初始化块            
    {        
        System.out.println(field);
        System.out.println("初始化块");      
    }                
    // 构造器        
    public InitialOrderTest() {
        System.out.println("构造器");   
    }                
    public static void main(String[] args) {
        new InitialOrderTest();            
    }        
}

 

运行以上代码,我们会得到如下的输出结果: 
静态变量  
静态初始化块  
变量  
初始化块  
构造器 
这与上文中说的完全符合。 
那么对于继承情况下又会怎样呢?我们仍然以一段测试代码来获取最终结果: 
Java代码 

Java代码 复制代码 收藏代码
  1. class Parent {               
  2.     // 静态变量           
  3.     public static String p_StaticField = "父类--静态变量";               
  4.     // 变量           
  5.     public String p_Field = "父类--变量";                   
  6.     // 静态初始化块               
  7.     static {           
  8.         System.out.println(p_StaticField);                   
  9.         System.out.println("父类--静态初始化块");               
  10.     }                   
  11.     // 初始化块               
  12.     {           
  13.         System.out.println(p_Field);           
  14.         System.out.println("父类--初始化块");               
  15.     }                   
  16.     // 构造器           
  17.     public Parent() {           
  18.         System.out.println("父类--构造器");               
  19.     }          
  20. }                   
  21. public class SubClass extends Parent {               
  22.     // 静态变量           
  23.     public static String s_StaticField = "子类--静态变量";               
  24.     // 变量           
  25.     public String s_Field = "子类--变量";               
  26.     // 静态初始化块               
  27.     static {           
  28.         System.out.println(s_StaticField);                   
  29.         System.out.println("子类--静态初始化块");           
  30.     }           
  31.     // 初始化块  http://ini.iteye.com/   
  32.     {           
  33.         System.out.println(s_Field);           
  34.         System.out.println("子类--初始化块");               
  35.     }                   
  36.     // 构造器           
  37.     public SubClass() {           
  38.         System.out.println("子类--构造器");               
  39.     }                   
  40.     // 程序入口           
  41.     public static void main(String[] args) {                   
  42.         new SubClass();              
  43.     }           
  44. }  
class Parent {            
    // 静态变量        
    public static String p_StaticField = "父类--静态变量";            
    // 变量        
    public String p_Field = "父类--变量";                
    // 静态初始化块            
    static {        
        System.out.println(p_StaticField);                
        System.out.println("父类--静态初始化块");            
    }                
    // 初始化块            
    {        
        System.out.println(p_Field);        
        System.out.println("父类--初始化块");            
    }                
    // 构造器        
    public Parent() {        
        System.out.println("父类--构造器");            
    }       
}                
public class SubClass extends Parent {            
    // 静态变量        
    public static String s_StaticField = "子类--静态变量";            
    // 变量        
    public String s_Field = "子类--变量";            
    // 静态初始化块            
    static {        
        System.out.println(s_StaticField);                
        System.out.println("子类--静态初始化块");        
    }        
    // 初始化块  http://ini.iteye.com/
    {        
        System.out.println(s_Field);        
        System.out.println("子类--初始化块");            
    }                
    // 构造器        
    public SubClass() {        
        System.out.println("子类--构造器");            
    }                
    // 程序入口        
    public static void main(String[] args) {                
        new SubClass();           
    }        
}

 运行一下上面的代码,结果马上呈现在我们的眼前: 

父类--静态变量  
父类--静态初始化块  
子类--静态变量  
子类--静态初始化块  
父类--变量  
父类--初始化块  
父类--构造器  
子类--变量  
子类--初始化块  
子类--构造器 
现在,结果已经不言自明了。大家可能会注意到一点,那就是,并不是父类完全初始化完毕后才进行子类的初始化, 
实际上子类的静态变量和静态初始化块的初始化是在父类的变量、初始化块和构造器初始化之前就完成了。 
那么对于静态变量和静态初始化块之间、变量和初始化块之间的先后顺序又是怎样呢? 
是否静态变量总是先于静态初始化块,变量总是先于初始化块就被初始化了呢?实际上这取决于它们在类中出现的先后顺序。 
我们以静态变量和静态初始化块为例来进行说明。 同样,我们还是写一个类来进行测试:    

Java代码 

Java代码 复制代码 收藏代码
  1. public class TestOrder {           
  2.     // 静态变量           
  3.     public static TestA a = new TestA();                        
  4.     // 静态初始化块               
  5.     static {           
  6.         System.out.println("静态初始化块");               
  7.     }                        
  8.     // 静态变量           
  9.     public static TestB b = new TestB();                   
  10.     public static void main(String[] args) {                   
  11.         new TestOrder();               
  12.     }           
  13. }                   
  14. class TestA {               
  15.     public TestA() {           
  16.         System.out.println("Test--A");               
  17.     }           
  18. }                   
  19. class TestB {               
  20.     public TestB() {           
  21.         System.out.println("Test--B");               
  22.     }           
  23. }  
public class TestOrder {        
    // 静态变量        
    public static TestA a = new TestA();                     
    // 静态初始化块            
    static {        
        System.out.println("静态初始化块");            
    }                     
    // 静态变量        
    public static TestB b = new TestB();                
    public static void main(String[] args) {                
        new TestOrder();            
    }        
}                
class TestA {            
    public TestA() {        
        System.out.println("Test--A");            
    }        
}                
class TestB {            
    public TestB() {        
        System.out.println("Test--B");            
    }        
}

 运行上面的代码,会得到如下的结果: 

Test--A  
静态初始化块  
Test--B 
大家可以随意改变变量a、变量b以及静态初始化块的前后位置,就会发现输出结果随着它们在类中出现的前后顺序而改变, 
这就说明静态变量和静态初始化块是依照他们在类中的定义顺序进行初始化的。同样,变量和初始化块也遵循这个规律。 
了解了继承情况下类的初始化顺序之后,如何判断最终输出结果就迎刃而解了。  

测试函数:  

Java代码 复制代码 收藏代码
  1. public class TestStaticCon {    
  2.     public static int a = 0;   
  3.       static {     
  4.         a = 10;   
  5.           System.out.println("父类的非静态代码块在执行a=" + a);    
  6.     }      
  7.     {     
  8.         a = 8;   
  9.           System.out.println("父类的非静态代码块在执行a=" + a);    
  10.     }   
  11.       public TestStaticCon() {    
  12.     
  13.         this("a在父类带参构造方法中的值:" + TestStaticCon.a); // 调用另外一个构造方法     
  14.         System.out.println(a);     
  15.         System.out.println("父类无参构造方法在执行a=" + a);    
  16.     }   
  17.       public TestStaticCon(String n) {     
  18.         System.out.println(n);     
  19.         System.out.println(a);   
  20.       }   
  21.       public static void main(String[] args) {     
  22.         TestStaticCon tsc = null;   
  23.           System.out.println("!!!!!!!!!!!!!!!!!!!!!");     
  24.         tsc = new TestStaticCon();    
  25.     }   
  26. }  
public class TestStaticCon { 
    public static int a = 0;
      static {  
        a = 10;
          System.out.println("父类的非静态代码块在执行a=" + a); 
    }   
    {  
        a = 8;
          System.out.println("父类的非静态代码块在执行a=" + a); 
    }
      public TestStaticCon() { 
 
        this("a在父类带参构造方法中的值:" + TestStaticCon.a); // 调用另外一个构造方法  
        System.out.println(a);  
        System.out.println("父类无参构造方法在执行a=" + a); 
    }
      public TestStaticCon(String n) {  
        System.out.println(n);  
        System.out.println(a);
      }
      public static void main(String[] args) {  
        TestStaticCon tsc = null;
          System.out.println("!!!!!!!!!!!!!!!!!!!!!");  
        tsc = new TestStaticCon(); 
    }
}

 运行结果: 

父类的非静态代码块在执行a=10 
!!!!!!!!!!!!!!!!!!!!! 
父类的非静态代码块在执行a=8 
a在父类带参构造方法中的值:10 


父类无参构造方法在执行a=8  

  
结论:静态代码块是在类加载时自动执行的,非静态代码块是在创建对象时自动执行的代码,不创建对象不执行该类的非静态代码块。且执行顺序为静态代码块------非静态代码块----构造函数。 
扩展:静态代码块  与  静态方法: 
一般情况下,如果有些代码必须在项目启动的时候就执行的时候,需要使用静态代码块,这种代码是主动执行的; 
需要在项目启动的时候就初始化,在不创建对象的情况下,其他程序来调用的时候,需要使用静态方法,这种代码是被动执行的.  
两者的区别就是:静态代码块是自动执行的;  静态方法是被调用的时候才执行的.  

作用:静态代码块可用来初始化一些项目最常用的变量或对象;静态方法可用作不创建对象也可能需要执行的代码

阿里笔试题:

求下面这段代码的输出:

Java代码 复制代码 收藏代码
  1. public class Test1 {   
  2.   
  3.     public static int k = 0;   
  4.     public static Test1 t1 = new Test1("t1");   
  5.     public static Test1 t2 = new Test1("t2");   
  6.     public static int i = print("i");   
  7.     public static int n = 99;   
  8.     public int j = print("j");   
  9.     {   
  10.         print("构造块");   
  11.     }   
  12.        
  13.     static{   
  14.         print("静态块");   
  15.     }   
  16.   
  17.     public Test1(String str){   
  18.         System.out.println((++k)+":"+str+"    i="+i+"    n="+n);   
  19.         ++i;++n;   
  20.     }   
  21.        
  22.     public static int print(String str){   
  23.         System.out.println((++k)+":"+str+"    i="+i+"    n="+n);   
  24.         ++n;   
  25.         return ++i;   
  26.     }   
  27.        
  28.     public static void main(String[] args) {   
  29.         // TODO Auto-generated method stub   
  30.         Test1 t = new Test1("init");   
  31.     }   
  32.   
  33. }  
public class Test1 {

	public static int k = 0;
	public static Test1 t1 = new Test1("t1");
	public static Test1 t2 = new Test1("t2");
	public static int i = print("i");
	public static int n = 99;
	public int j = print("j");
	{
		print("构造块");
	}
	
	static{
		print("静态块");
	}

	public Test1(String str){
		System.out.println((++k)+":"+str+"    i="+i+"    n="+n);
		++i;++n;
	}
	
	public static int print(String str){
		System.out.println((++k)+":"+str+"    i="+i+"    n="+n);
		++n;
		return ++i;
	}
	
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Test1 t = new Test1("init");
	}

}

 运行结果:

1:j    i=0    n=0
2:构造块    i=1    n=1
3:t1    i=2    n=2
4:j    i=3    n=3
5:构造块    i=4    n=4
6:t2    i=5    n=5
7:i    i=6    n=6
8:静态块    i=7    n=99
9:j    i=8    n=100
10:构造块    i=9    n=101
11:init    i=10    n=102

分享到:
评论

相关推荐

    java类中静态域、块,非静态域、块,构造函数的初始化顺序

    java类中静态域、块,非静态域、块,构造函数的初始化顺序,通过本例子,能够清楚了解到java类的初始化顺序

    java 面试题 总结

    不能有抽象构造函数或抽象静态方法。Abstract 类的子类为它们父类中的所有抽象方法提供实现,否则它们也是抽象类为。取而代之,在子类中实现该方法。知道其行为的其它类可以在类中实现这些方法。 接口(interface)...

    java编程规范(第三版)

    8.7 静态初始化语句 181 8.8 构造函数声明 182 8.9 枚举 189 第9章 接口 197 9.1 接口声明 197 9.2 接口成员 200 9.3 字段(常量)声明 201 9.4 抽象方法声明 203 9.5 成员类型声明 205 9.6 注释类型 206 ...

    Java2核心技术.part5

    3.10.2数组初始化器以及匿名数组 3.10.3数组拷贝 3.10.4命令行参数 3.10.5数组排序 3.10.6多维数组 3.10.7不规则数组 第4章对象与类 4.1面向对象程序设计概述 4.1.1 OOP词汇表 4.1.2对象 ...

    Java并发编程实战

    8.3.5 在调用构造函数后再定制ThreadPoolExecutor147 8.4 扩展 ThreadPoolExecutor148 8.5 递归算法的并行化149 第9章 图形用户界面应用程序156 9.1 为什么GUI是单线程的156 9.1.1 串行事件处理157 9.1.2 ...

    Java2核心技术.part3

    3.10.2数组初始化器以及匿名数组 3.10.3数组拷贝 3.10.4命令行参数 3.10.5数组排序 3.10.6多维数组 3.10.7不规则数组 第4章对象与类 4.1面向对象程序设计概述 4.1.1 OOP词汇表 4.1.2对象 ...

    Java2核心技术.part1

    3.10.2数组初始化器以及匿名数组 3.10.3数组拷贝 3.10.4命令行参数 3.10.5数组排序 3.10.6多维数组 3.10.7不规则数组 第4章对象与类 4.1面向对象程序设计概述 4.1.1 OOP词汇表 4.1.2对象 4.1.3类...

    Java2核心技术.part6

    3.10.2数组初始化器以及匿名数组 3.10.3数组拷贝 3.10.4命令行参数 3.10.5数组排序 3.10.6多维数组 3.10.7不规则数组 第4章对象与类 4.1面向对象程序设计概述 4.1.1 OOP词汇表 4.1.2对象 ...

    Java2核心技术.part4

    3.10.2数组初始化器以及匿名数组 3.10.3数组拷贝 3.10.4命令行参数 3.10.5数组排序 3.10.6多维数组 3.10.7不规则数组 第4章对象与类 4.1面向对象程序设计概述 4.1.1 OOP词汇表 4.1.2对象 ...

    Java2核心技术.part2

    3.10.2数组初始化器以及匿名数组 3.10.3数组拷贝 3.10.4命令行参数 3.10.5数组排序 3.10.6多维数组 3.10.7不规则数组 第4章对象与类 4.1面向对象程序设计概述 4.1.1 OOP词汇表 4.1.2对象 ...

    C++复习资料之系列

    ( c ) 可以在程序的任何地方 (d) 必须在其它函数中间 2.用C++语言编制的源程序要变为目标程序必须要经过( d )。 (a) 解释 (b) 汇编 (c) 编辑 (d) 编译 3.C++程序基本单位是( c )。 (a) 数据 (b) 字符 (c) ...

    c#学习笔记——学习心得

    向方法传递结构时,是通过传值方式传递的,结构实例化可以不用new,结构可以声明构造函数,但必须带参数,且声明的构造函数是用来对成员初始化的,必须包含每个字段。结构不能从另一个结构或类继承而来,但可以实现...

    javaSE代码实例

    15.4.3 匿名内部类的初始化 337 15.4.4 匿名内部类作用的体现 337 15.5 理解内部类 339 15.6 内部接口 340 15.6.1 定义在类中的内部接口 340 15.6.2 定义在接口中的内部接口 341 15.7 小结 342 第16...

    超爽的自学课件(java)

    在这一章的最后,我们将更贴近地观察初始化过程:自动成员初始化、指定成员初始化、初始化的顺序、static(静态)初始化以及数组初始化等等。 <br>(5) 第5章:隐藏实现过程 本章要探讨将代码封装到一起的方式...

    超级有影响力霸气的Java面试题大全文档

    不能有抽象构造函数或抽象静态方法。Abstract 类的子类为它们父类中的所有抽象方法提供实现,否则它们也是抽象类为。取而代之,在子类中实现该方法。知道其行为的其它类可以在类中实现这些方法。  接口(interface...

    Visual C++ 2005入门经典--源代码及课后练习答案

    7.4.4 在构造函数中使用初始化列表 320 7.5 类的私有成员 320 7.5.1 访问私有类成员 323 7.5.2 类的友元函数 324 7.5.3 默认复制构造函数 326 7.6 this指针 328 7.7 类的const对象 331 7.7.1 类的...

    Visual C++ 2010入门经典(第5版)--源代码及课后练习答案

    7.4.4 在构造函数中使用初始化列表 316 7.4.5 声明显式的构造函数 317 7.5 类的私有成员 318 7.5.1 访问私有类成员 320 7.5.2 类的友元函数 321 7.5.3 默认复制构造函数 323 7.6 this指针 325 7.7 类的const...

    Java开发实战1200例(第1卷).(清华出版.李钟尉.陈丹丹).part3

    实例085 域的默认初始化值 106 实例086 编写同名的方法 107 实例087 构造方法的应用 108 5.2 修饰符的使用 109 实例088 单例模式的应用 109 实例089 祖先的止痒药方 110 实例090 统计图书的销售量 111 实例091 汉诺...

Global site tag (gtag.js) - Google Analytics