Java是面向对象的编程语言,类与对象可以说是构建思想里的核心。不说了,先上图
这张图是最基本的,将类与对象的组成部分作了个梳理,当然还不够全面,没有将接口、继承和多态等等都写进去。但是,把这张图弄懂,是后续的添加抽象类、final类以及继承中的父子类等知识的根本。
在这里,我觉得没有必要把所有的细节都写出来,例如静态只能调用静态,构造函数在创建对象时进行初始化等。只要自己学了一遍,这些细节基本都知道了。
我来谈谈学到类与对象时的一个关键点:初始化顺序。
先假设一个类,如图中一样各组成都有,那么他的初始化顺序是
- 初始化类变量(即static修饰的成员变量),并未赋值。不管写的位置在哪里,只要是类变量,系统总会先找到它进行变量初始化。
- 执行静态代码块和类变量定义式,两者根据写的位置来决定先后,先写先执行。其实从某种角度上看,可以把类变量定义赋值视为两部分:一部分是定义变量,一部分赋值。而这个赋值部分可以看做是一个静态代码块。两个静态代码块的执行顺序自然是看写的位置的先后了。
- 初始化实例变量(即未被static修饰的成员变量),并未赋值。同样的,不管写的位置在哪里,在创建对象时执行到这步时,系统总会找到它进行变量初始化。
- 执行构造代码块和实例变量定义赋值式,两者同样根据写的位置先后来决定执行顺序先后,同样可以按2中所写来理解。但是,这里要注意的就是构造代码块是可以调用静态变量的,实例变量定义赋值式可以看做是只对实例变量进行赋值的构造代码块。
- 执行构造函数。构造函数同样可以调用静态变量和实例变量。
初始化结束。
这里说明一点:这是初始化顺序,不等同于语句程序的执行过程(毕老师的视频里有个很详细的例子讲这个执行过程,不知道的一定要去看)。因此在上面的初始化顺序里没有成员函数(静态或者非静态都没有),这是因为成员函数都是调用了才执行,虽然静态函数已经被加载进了方法区,但初始化过程中并没有执行过。
关于这个初始化顺序,其实一句话可以概括:
先初始化类变量然后赋值,再初始化实例变量然后赋值。
由于静态代码块可以调用静态变量,构造代码块和构造函数可以调用实例变量和静态变量,这块很容易来个看似复杂的代码,将一个变量变来变去的,弄明白这个初始化顺序就会解决很快了。
接下来,看几个例子来验证下:
第一个:
public class JustForTest {
public static void main(String[] args) {
Car c=new Car();
sop("i="+c.i);
}
static void sop(Object obj){
System.out.println(obj);
}
}
class Car{
static int i=1; //定义赋值
static { //静态代码块
i=4;
}
}
运行结果为:i=4.
只改写Car的内部,让静态代码块和静态变量的定义赋值互换位置,其他保持不变:
class Car{
static { //静态代码块
i=4;
}
static int i=1; //定义赋值
}
运行结果为:i=1.
最后来个综合点的,把Car再改写一下:
class Car{
static int i=1; //静态变量定义赋值
Car(){ //构造函数
i=2;
}
static { //静态代码块
i=4;
}
{ //构造代码块
i=3;
}
}
运行结果是:i=2.
按初始化顺序,构造函数是最后初始化的。