当一个类使用new关键字来建立新的对象的时候,好比Person per = new Person();JVM根据Person()寻找匹配的类,而后找到这个类相匹配的构造方法,这里是无参构造,若是程序中没有给出任何构造方法,则JVM默认会给出一个无参构造。当建立一个对象的时候必定对调用该类的构造方法,构造方法就是为了对对象的数据进行初始化。JVM会对给这个对象分配内存空间,也就是对类的成员变量进行分配内存空间,若是类中在定义成员变量就赋值的话,就按初始值进行运算,若是只是声明没有赋初始值的话,JVM会按照规则自动进行初始化赋值。而成员方法是在对象调用这个方法的时候才会从方法区中加载到栈内存中,用完就当即释放内存空间。java
笔者认为学到类与对象时的一个关键点:初始化顺序。
先假设一个类,如图中同样各组成都有,那么他的初始化顺序是函数
一、初始化类变量(即static修饰的成员变量),并未赋值。无论写的位置在哪里,只要是类变量,系统总会先找到它进行变量初始化。
二、执行静态代码块和类变量定义式,二者根据写的位置来决定前后,先写先执行。其实从某种角度上看,能够把类变量定义赋值视为两部分:一部分是定义变量,一部分赋值。而这个赋值部分能够看作是一个静态代码块。两个静态代码块的执行顺序天然是看写的位置的前后了。
三、初始化实例变量(即未被static修饰的成员变量),并未赋值。一样的,无论写的位置在哪里,在建立对象时执行到这步时,系统总会找到它进行变量初始化。
四、执行构造代码块和实例变量定义赋值式,二者一样根据写的位置前后来决定执行顺序前后,一样能够按2中所写来理解。可是,这里要注意的就是构造代码块是能够调用静态变量的,实例变量定义赋值式能够看作是只对实例变量进行赋值的构造代码块。
五、执行构造函数。构造函数一样能够调用静态变量和实例变量。
初始化结束。 code
这里说明一点:这是初始化顺序,不等同于语句程序的执行过程(毕老师的视频里有个很详细的例子讲这个执行过程,不知道的必定要去看)。所以在上面的初始化顺序里没有成员函数(静态或者非静态都没有),这是由于成员函数都是调用了才执行,虽然静态函数已经被加载进了方法区,但初始化过程当中并无执行过。
关于这个初始化顺序,其实一句话能够归纳:
先初始化类变量而后赋值,再初始化实例变量而后赋值。
因为静态代码块能够调用静态变量,构造代码块和构造函数能够调用实例变量和静态变量,这块很容易来个看似复杂的代码,将一个变量变来变去的,弄明白这个初始化顺序就会解决很快了。 视频
接下来,看几个例子来验证下: 对象
第一个:内存
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.class
只改写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.
按初始化顺序,构造函数是最后初始化的。