实例初始化程序块用于初始化实例数据成员。 它在每次创建类的对象时运行。实例变量的初始化可以是直接的,但是可以在初始化实例初始化块中的实例变量时执行额外的操作。
什么是实例初始化块的使用,我们可以直接分配一个值在实例数据成员? 例如:
class Bike{ int speed=100; }
为什么要使用实例初始化块?
假设在赋值给实例数据成员时必须要执行一些操作。 如:使用for
循环来填充复杂数组或错误处理等。
实例初始化程序块的示例
下面来看看实例初始化块执行初始化的简单例子。
class Bike7 { int speed; Bike7() { System.out.println("speed is " + speed); } { speed = 100; } public static void main(String args[]) { Bike7 b1 = new Bike7(); Bike7 b2 = new Bike7(); } }
执行上面代码得到以下结果 -
speed is 100 speed is 100
在Java中有三个地方可以执行实例初始化块操作:
- 方法
- 构造函数
- 代码块
实例初始化块或构造函数哪个先调用?
看看下面代码就知道了。
class Bike8 { int speed; Bike8() { System.out.println("constructor is invoked"); } { System.out.println("instance initializer block invoked"); } public static void main(String args[]) { Bike8 b1 = new Bike8(); Bike8 b2 = new Bike8(); } }
执行上面代码得到以下结果 -
instance initializer block invoked constructor is invoked instance initializer block invoked constructor is invoked
在上面的例子中,似乎首先调用实例初始化块,但是可惜不是。 实例初始化块在对象创建时被调用。 java编译器在第一个语句super()
之后的构造函数中复制实例初始化块。 首先,调用构造函数。 让我们通过下面的图来理解它:
注意:java编译器在每个构造函数中会自动复制实例初始化块的代码。
实例初始化块的规则:
实例初始化块主要有三个规则。 它们如下:
- 在创建类的实例时创建实例初始化程序块。
- 在父类构造函数被调用之后(即在
super()
构造函数调用之后)调用实例初始化块。 - 实例初始化程序块按它们显示的顺序排列。
在super()
之后调用的实例初始化块的程序示例:
class A { A() { System.out.println("parent class constructor invoked"); } } class B2 extends A { B2() { super(); System.out.println("child class constructor invoked"); } { System.out.println("instance initializer block is invoked"); } public static void main(String args[]) { B2 b = new B2(); } }
上面代码执行结果如下 -
parent class constructor invoked instance initializer block is invoked child class constructor invoked
实例块的另一个示例
class A { A() { System.out.println("parent class constructor invoked"); } } class B3 extends A { B3() { super(); System.out.println("child class constructor invoked"); } B3(int a) { super(); System.out.println("child class constructor invoked " + a); } { System.out.println("instance initializer block is invoked"); } public static void main(String args[]) { B3 b1 = new B3(); B3 b2 = new B3(10); } }
上面代码执行结果如下 -
parent class constructor invoked instance initializer block is invoked child class constructor invoked parent class constructor invoked instance initializer block is invoked child class constructor invoked 10