面试系列(一):代码的执行顺序
开始这个系列是因为想总结一下面试中比较常见的考点。。不喜勿喷~~~~~~
1、静态代码块、构造代码块、普通代码块和构造函数的执行顺序
静态代码块:在java中使用static关键字声明的代码块。静态块用于初始化类,为类的属性初始化。每个静态代码块只会执行一次。由于JVM在加载类时会执行静态代码块,所以静态代码块先于主方法执行。
构造代码块:直接在类中定义且没有加static关键字的代码块称为{}构造代码块。构造代码块在创建对象时被调用,每次创建对象都会被调用,并且构造代码块的执行次序优先于类构造函数。
普通代码块:在方法或语句中出现的{}就称为普通代码块。普通代码块和一般的语句执行顺序由他们在代码中出现的次序决定--“先出现先执行”
构造函数:也称构造器,和类同名,没有返回类型。(void也没有)
那么他们的执行顺序如何呢?用一个小例子可以了解~
class A { { System.out.println("我是构造代码块======="); } static { System.out.println("我是静态代码块======="); } A() { System.out.println("我是构造函数========="); } public void test() { System.out.println("我是普通代码块========="); } public static void main(String[] args) { System.out.println("我是main普通代码块========="); } }
我们先运行A本身:
我是静态代码块=======
我是main普通代码块=========
我们再用另一个类调用下:
A a = new A(); a.test(); System.out.println(); A a1 = new A(); a1.test();
打印结果:
我是静态代码块=======
我是构造代码块=======
我是构造函数=========
我是普通代码块=========
我是构造代码块=======
我是构造函数=========
我是普通代码块=========
由此可见:每个类的静态代码块只会执行一次,无论你初始化多少次;而构造代码块、构造函数是每初始化一次就执行一次的。
执行顺序:静态代码块>构造代码块>构造方法>main方法(构造代码块和构造方法只有在被初始化的时候才会调用)
2、构造函数与继承
每个类都有一个默认的构造函数(不带参数),可以不用定义;但是当你重载了一个带参的构造函数,则默认的构造函数将会被覆盖,如果你想同时拥有默认的构造器和重载的带参的构造器,则默认构造器必须定义出来。
而构造函数的继承也是常见考点~也是比较蛋疼的地方。。
class Aoo { public Aoo() { System.out.println("Aoo 默认构造器"); } public Aoo(int i) { System.out.println("Aoo 带参构造器:" + i); } public void test() { System.out.println("Aoo 的test方法"); } public void test(int i) { System.out.println("Aoo 的test带参方法:" + i); } } class Boo extends Aoo{ public Boo() { System.out.println("Boo 默认构造器"); } public Boo(int i) { System.out.println("Boo 带参构造器:" + i); } public void test() { System.out.println("Boo 的test方法"); } public void test(int i) { System.out.println("Boo 的test带参方法:" + i); } }
我们再调用:
Boo boo = new Boo();//或Aoo boo = new Boo(); 结果都一样 boo = new Boo(1); boo.test(); boo.test(1);
打印结果:
Aoo 默认构造器
Boo 默认构造器
Aoo 默认构造器
Boo 带参构造器:1
Boo 的test方法
Boo 的test带参方法:1
说明:
1)子类永远会默认继承父类的默认(不带参)的构造函数,子类每初始化一次,必然会调用一次父类的不带参的默认构造器。
2)当子类和父类都有相同方法声明的函数时,具体调用子类还是父类的方法是看初始化的谁,例如本例中是new B,而与变量类型无关;
3、重载和重写
重载:在一个类里,方法名相同,方法声明不同(参数声明不同,返回值类型也可能不同);
重写:涉及继承,子类和父类拥有相同的方法名和方法声明,返回值类型必须是父类对应方法返回类型的子类。
重载是一个类里的多态性,重写是子类和父类的多态性体现~
我们先定义2个父子类:
class Super { public void t() { System.out.println("Super t()"); } } class Sub extends Super { public void t() { System.out.println("Sub t()"); } }
这里Sub子类重写了父类的t()方法,
我们再定义一个类,里面有重载的2个方法:
class Goo { public void test(Super obj) { System.out.println("test Super"); obj.t(); } public void test(Sub obj) { System.out.println("test Sub"); obj.t(); } }
再调用:
Goo goo = new Goo(); Super obj = new Sub(); goo.test(obj);
打印结果:
test Super
Sub t()
这说明:
1)Goo类里有重载的test方法,goo.test(obj);里的参数obj的类型是Super,所以调用的是Goo的第一个test方法;所以打印出:test Super
2)虽然Super obj = new Sub();但调用涉及重写的方法时,看的是真正实例化的类,这里虽然用父类Super定义,但实例化的是子类Sub,所以调用的子类的t()方法~