静态优先,父类优先。
(1) 父类静态代码块(包括静态初始化块,静态属性,但不包括静态方法)
(2) 子类静态代码块(
包括静态初始化块,静态属性,但不包括静态方法
)
实例化后
(3) 父类非静态代码块(
包括非静态初始化块,非静态属性
)
(4) 父类构造函数
(5) 子类非静态代码块
(
包括非静态初始化块,非静态属性
)
(6) 子类构造函数
其中:类中静态块按照声明顺序执行,并且(1)和(2)不需要调用new类实例的时候就执行了(意思就是在类加载到方法区的时候执行的)
2.其次,需要理解子类覆盖父类方法的问题,也就是
方法重写实现多态
问题。
Base b = new Sub();
它为多态的一种表现形式,声明是Base,实现是Sub类,
理解为
b
编译时表现为Base类特性,运行时表现为Sub类特性。
一个类可以使用不包含在任何方法体中的静态代码块,当类被载入时,静态代码块被执行,且只被执行一次,静态块常用来执行类属性的初始化。
一.类加载时,变量的初始化顺序:
1.首先,给静态成员变量分配内存空间,进行默认初始化
(整型为0,浮点型为0.0,布尔型为false,字符型为'\u0000',引用型为null)
2.其次,执行静态成员变量的初始化操作
--静态成员的初始化,包括两种: 声明时直接初始化和静态代码块
--执行顺序为:在代码中的出现的顺序(声明的顺序)
注意: 1和2的动作,在类加载时只执行一次!!!!!!!
如果创建了对象,则在堆中给类的实例分配内存空间,
3.首先,对实例变量,进行默认初始化
(整型为0,浮点型为0.0,布尔型为false,字符型为'\u0000',引用型为null)
4.其次,执行实例变量的初始化操作
--实例变量的初始化,使用前2种初始化方式: 声明时直接初始化和代码块
--执行顺序为:在代码中的出现的顺序(声明的顺序)
5.最后,执行构造函数(实例变量的第3中初始化方式)
二.继承时,变量的初始化顺序:
1.当类第一次使用时,JVM就会加载该类,如果该类存在父类,那么就先加载父类,这是一个递归过程,直到Object为止.
在类加载中,首先进行静态成员变量按照默认值进行初始化,
然后按照在类中声明的顺序执行静态代码块和静态变量的显示初始化.
这个过程从父类到子类,并且只会执行一次!!!
2.当父类与子类的静态代码初始化完成后,如果创建了类的对象,
在初始化子类前,会先对其父类的实例变量进行默认初始化,
然后按照在类中的声明顺序来执行代码块与实例变量的显示初始化,
最后调用父类的构造函数,这也是一个递归过程,直到Object类为止.
(这个过程在每次创建对象时,都会执行!!!)
在特殊情况如在父类构造方法调用自身函数,再由子类重写
输出null
其中
因为,在编译阶段,编译器只知道对象的静态类型,而不知道实际类型,所以在class文件中只能确定要调用父类的方法。但是在执行时却会判断对象的实际类型。如果实际类型实现这个方法,则直接调用,如果没有实现,则按照继承关系从下往上一次检索,只要检索到就调用,如果始终没有检索到,则抛异常
new Sub();在创造派生类的过程中首先创建基类对象,然后才能创建派生类。
(声明为父类,所以编译阶段仅知道父类)
创建基类即默认调用Base()方法,在方法中调用callName()方法
(因为执行时会判断对象实际类型为子类)
,由于派生类中存在此方法,则被调用的callName()方法是派生类中的方法,此时派生类还未构造,所以变量baseName的值为null
(因为没有构建所以只知道其静态形态,常量内容的分配是动态的没有构建所以值为null)