本文转自此处
源代码:
public class Teststaticblock { public Teststaticblock() { this("second"); System.out.println("begin constructor"); System.out.println(s_a); //111 System.out.println(s_b); //222 System.out.println(c); //333 System.out.println(d); //d的值为什么是4,却不是444?? //因为程序会按照初始化块和实例Field生命语句的先后顺序来执行, //先赋给d的值444会被后赋给d的值4覆盖掉,所以d的最终值为4. // this("second");//call to this must be first statement in constructor s_a=1111; s_b=2222; c=3333; d=4444; System.out.println(s_a); //1111 System.out.println(s_b); //2222 System.out.println(c); //3333 System.out.println(d); //4444 System.out.println("end constructor"); } public Teststaticblock(String s) { System.out.println("begin second constructor"); System.out.println("end second constructor"); } public static void main(String args[]) { System.out.println("begin main"); System.out.println(s_a); System.out.println(s_b); //s_b的值为什么是2,却不是22?? //因为程序会按照静态初始化块和静态Field声明语句的先后顺序来执行,先赋给s_b的值22 //会被后赋给s_b的值2覆盖掉,所以s_b最终的值为2. // System.out.println(c); //non-static variable c cannot be referenced from a static context // System.out.println(d); //non-static variable c cannot be referenced from a static context s_a=11111; s_b=22222; // c=33333;//non-static variable c cannot be referenced from a static context // d=44444;//non-static variable c cannot be referenced from a static context System.out.println(s_a); //11111 System.out.println(s_b); //22222 // System.out.println(c); //non-static variable c cannot be referenced from a static context // System.out.println(d); //non-static variable c cannot be referenced from a static context System.out.println("before new class object"); Teststaticblock t = new Teststaticblock(); System.out.println("end new class object"); System.out.println(s_a);//1111 System.out.println(s_b); //2222 // System.out.println(c); //non-static variable c cannot be referenced from a static context // System.out.println(d); //non-static variable c cannot be referenced from a static context s_a=111111; s_b=222222; // c=333333;//non-static variable c cannot be referenced from a static context // d=444444;//non-static variable c cannot be referenced from a static context System.out.println(s_a); //111111 System.out.println(s_b); //222222 // System.out.println(c); //non-static variable c cannot be referenced from a static context // System.out.println(d); //non-static variable c cannot be referenced from a static context System.out.println("end main"); } static int s_a=1; int c=3; { System.out.println("begin block"); System.out.println(s_a); //11111 System.out.println(s_b); //22222 System.out.println(c); //3 // System.out.println(d);//illegal forward reference s_a=111; s_b=222; c=333; d=444; System.out.println(s_a); //111 System.out.println(s_b); //222 System.out.println(c); //333 // System.out.println(d);//illegal forward reference System.out.println("end block"); } static { System.out.println("begin static block"); //begin static block System.out.println(s_a);//1 //为什么不能输出s_b的值,却可以给s_b赋值?? // System.out.println(s_b);//illegal forward reference // System.out.println(c); //non-static variable c cannot be referenced from a static context // System.out.println(d); //non-static variable c cannot be referenced from a static context s_a=11; s_b=22; System.out.println(s_a); //11 // System.out.println(s_b);//illegal forward reference // System.out.println(c); //non-static variable c cannot be referenced from a static context // System.out.println(d); //non-static variable c cannot be referenced from a static context System.out.println("end static block"); } int d=4; static int s_b=2; }
运行结果:
/* begin static block 1 11 end static block begin main 11 2 11111 22222 before new class object begin block 11111 22222 3 111 222 333 end block begin second constructor end second constructor begin constructor 111 222 333 4 1111 2222 3333 4444 end constructor end new class object 1111 2222 111111 222222 end main */
结果分析:
1、在类第一次加载时候,会执行静态成员变量(静态Field)初始化语句和静态初始化块(用static{}包含的部分)。
注意:
a、不管静态成员变量声明语句的实际位置在哪儿,当第一次加载类的时候都会首先对它初始化为缺省值(0,false,null等)。
b、即使静态成员变量声明中使用了显式初始化语句(比如:int x=3),第一次加载类的时候也会先把它初始化为缺省值(此时x为0),然后再按照下面说的要点c来执行赋值语句(x=3)。
c、对于静态成员变量的显式初始化语句和静态初始化块,按照在类中代码出现的先后顺序执行。
因此,在上面的例子程序中,我们看到
static int s_a=1;
static
{
s_a=11;
s_b=22;
}
static int s_b=2;
对s_a,s_b会有不同的效果。
类加载时候,s_a,s_b都被初始化为0,然后由于依照代码顺序执行了s_a=1;s_a=11;s_b=22;s_b=2;
结果s_a、s_b分别变成了11和2。
2、当构造类实例时候,会先对实例成员变量初始化为缺省值,然后执行初始化块(用{}括起来的部分),然后执行构造方法。其中:
a、如同1中一样,如果有实例成员变量的显式初始化语句,程序仍然是先将该成员变量初始化为缺省值,然后按照代码在类中出现的先后顺序执行初始化语句或者初始化块。
如果初始化块位置在初始化语句前面,即使它改变了该成员变量的值,也会被随后执行的初始化语句改回去。
b、在进入构造方法后,如果构造方法第一句是使用this(...)调用另一构造方法的话,则先执行另一构造方法,然后再执行本构造方法的方法体。这种用法必须让this(...)位于第一句。
相关推荐
构造函数主要用于初始化对象的状态,而静态块则是在类加载到内存中时执行的一段代码,通常用于类级别的初始化。 ### 构造函数与静态块的基本概念 构造函数是一种特殊的方法,它与类同名,并没有返回类型,其主要...
静态初始化块常用于设置类级别的变量,或者执行只应执行一次的初始化操作。例如,如果一个类需要在程序运行前设置一个常量数组,静态初始化块就是合适的选择。 ```java public class InitFiledBlockStatic { ...
System.out.println("父类--静态初始化块"); } // 初始化块 { System.out.println(p_Field); System.out.println("父类--初始化块"); } // 构造器 public Parent() { System.out.println(...
- **非静态块(实例初始化块)**:在每次创建类的新实例时执行的代码块,用于初始化非静态域。 - **构造函数**:用于初始化新创建的对象的方法。 #### 二、初始化顺序 按照 Java 的规定,以下列出了类初始化的基本...
初始化块可以在类定义中定义,并且在构造函数调用之前执行。初始化块的执行顺序同样遵循它们在类体中出现的顺序。 #### 示例代码分析 为了更好地理解类继承初始化顺序的概念,下面通过一段示例代码来进行详细分析...
3. **初始化块与成员变量初始化的顺序**:如果静态成员变量定义与静态初始化块同时存在,则先执行静态成员变量的初始化;同理,对于非静态成员变量也是如此。 掌握这些原则可以帮助开发者更准确地控制类的初始化...
这意味着如果一个全局变量依赖于其他全局变量的初始化结果,必须确保这些依赖的变量已完成静态初始化。 初始化顺序方面,同一编译单元内的全局变量按照声明顺序进行初始化,但不同编译单元间的全局变量的初始化顺序...
类的初始化涉及到多个方面,包括静态成员变量、实例成员变量、静态初始化块、实例初始化块以及构造函数等。本文将详细探讨Java中类的初始化过程及其顺序,并通过具体的代码示例来帮助理解这一过程。 #### 二、基础...
总之,Java代码的初始化顺序是类加载的必然过程,涉及到静态和实例初始化块、构造函数、成员变量初始化以及继承关系的影响。这个demo是学习和理解这些概念的重要工具,通过实际操作可以加深对Java内存管理和对象生命...
静态构造函数是用于初始化静态数据成员或者执行静态初始化的特殊方法。它在程序首次引用该类的静态成员时执行,并且只会执行一次。在 `Cow` 类的示例中,静态构造函数 `static Cow()` 被用来初始化静态变量 `count`...
需要注意的是,静态初始化块只在类被加载时执行一次,而实例初始化块(构造函数)则会在每次创建类的实例时执行。静态变量的初始化是在类加载时完成的,而实例变量的初始化则在对象创建时进行。 在实际开发中,对...
在 Java 中,静态变量的初始化过程可以被分为三个阶段:静态初始化、非静态初始化和执行构造函数。静态初始化是指在类加载时对静态变量的初始化。非静态初始化是指在实例创建时对实例变量的初始化。执行构造函数是指...
通过这两个示例,我们可以看到,非静态成员变量的初始化顺序是:成员变量初始化语句>成员变量初始化块>构造函数。 而对于静态成员变量,它们是在类加载时初始化的。例如,我们可以定义一个静态成员变量static ...
- **父类静态成员赋值和静态初始化块**:当类首次被加载时,其父类的静态成员和静态初始化块将被优先执行。这是因为在继承体系中,父类的概念必须先于子类存在。 - **子类静态成员赋值和静态初始化块**:接着,子类...
构造函数是在创建对象时执行,而静态初始化块是在类加载时执行。两者不会相互影响,它们各自负责不同的初始化任务。 4. **多线程环境**:如果多个线程同时加载同一个类,静态初始化块会被同步执行,确保线程安全。...
2. **静态初始化**:每个类的静态成员变量和静态初始化块(如果有的话)会在类加载时被初始化。在`Base`类中,静态变量`a`首先被赋值为10,然后执行静态初始化块,打印出"Static Init Base 10"。接下来,JVM继续加载...
2. 静态初始化代码块中对类变量指定初值。 这里类变量初始化的时机为两处,而没有在构造器中这一说。这也很容易理解,因为类变量是类加载的时候执行的初始化,且只执行一次,而调用构造器是对象实例化执行的,每...
MyClass(int n) : normalVar(n) {} // 构造函数初始化普通变量 void printVars() { std::cout ; std::cout ; } static void setStatic(int s) { staticVar = s; } // 静态成员函数 }; int MyClass::static...
1.只能在构造函数初始化列表初始化的成员变量的类型? a.const成员变量 b.引用类型的成员变量 c.static不能在初始化列表中进行初始化 d.类成员变量中有自定义类型的变量最好在初始化列表中进行初始化 2.初始...
接着,我们讨论非静态代码块,也称为实例初始化块。这种代码块在创建类的每个新实例时执行,用于初始化对象的状态。非静态代码块可以包含任意Java代码,但通常用于设置实例变量的初始值。比如: ```java public ...