静态变量或静态代码块(先后与代码书写顺序有关)-》初始化代码块或普通变量(顺序与代码书写顺序有关)-》构造函数
/*
* 几大原则
* 一、静态成员变量(Static)
* 1、静态成员变量为类变量,所有对象共享同一内存空间
* 2、静态成员变量的声明和定义仅在首次加载类时执行一次
* 3、首次加载类时首先对所有静态成员变量根据类型默认赋初值,然后再对有右值的附右值
* 二、静态初始块
* 1、静态初始化块仅在首次加载类时执行一次
* ······多个静态成员变量与静态始化快参照出现顺序先后执行······
* 三、动态成员变量
* 1、动态成员变量定义在每次实例化对象时在构造函数之前执行
* 四、动态初始化块
* 1、动态初始化块在每次实例化对象时在构造函数之前执行
* ······多个动态成员变量与动态初始化块参照出现顺序先后执行······
* 总结:总的来说,在不涉及继承的前提下,当首次加载类时,按照如下顺序执行
* 1、按照出现顺序先后执行静态成员变量定义与静态初始化块
* 2、按照出现顺序先后执行动态成员变量定义与动态初始化块
* 3、执行构造函数
* 再次实例化对象时只执行第2、3步即可
*
* ············成员变量与定义与初始化块先于构造函数执行·········
* 五、当涉及到继承时,按照如下顺序执行
* 1、执行父类的静态成员变量定义与静态初始化块,执行子类的静态成员变量定义与静态初始化块
* 2、执行父类的非静态成员变量定义与动态初始化块,执行父类构造方法
* 3、执行子类的非静态成员变量定义与动态初始化块,执行子类构造方法
* 另:父类构造方法中用到的方法如果已被子类重写,那么在构造子类对象时在调用父类构造函数中使用子类重写的方法
*/
非静态的变量 在实例化对象时初始化 陷入死循环 所以内存溢出
结合上一个例子看出,初始化块{ }和非静态变量是按顺序执行,当把ADemo instance=new ADemo();放到初始化块{}后面时 ,{}会执行然后遇到new之后再从上到下顺序执行 遇到{}之后再进入new一直循环执行直到溢出
类会在首次被“主动使用”时执行初始化,为类(静态)变量赋予正确的初始值。在Java代码中,一个正确的初始值是通过类变量初始化语句或者静态初始化块给出的。而我们这里所说的主动使用 包括:
- 创建类的实例
- 调用类的静态方法
- 使用类的非常量静态字段
- 调用Java API中的某些反射方法
- 初始化某个类的子类
- 含有main()方法的类启动时【也就是上面的main 方法 中为空,即使不调用ADemo.id,也会执行static代码块】