次の3つの結論を理解する
class A {
private static int numA;
private int numA2;
static {
System.out.println("A的静态字段 : " + numA);
System.out.println("A的静态代码块");
}
{
System.out.println("A的成员变量 : " + numA2);
System.out.println("A的非静态代码块");
}
public A() {
System.out.println("A的构造器");
}
public A(int n) {
System.out.println("A的有参构造");
this.numA2 = n;
}
}
class B extends A {
private static int numB;
private int numB2;
static {
System.out.println("B的静态字段 : " + numB);
System.out.println("B的静态代码块");
}
{
System.out.println("B的成员变量 : " + numB2);
System.out.println("B的非静态代码块");
}
public B() {
System.out.println("B的构造器");
}
public B(int n) {
System.out.println("B的有参构造");
this.numB2 = n;
}
}
public class Test4{
public static void main(String[] args) {
B anotherB = new B(1);// 思考有参构造的输出结果
}
}
サブクラスBのパラメーター化された構成が呼び出されると、親クラスのコンストラクターが最初に実行されることは確かですが、どのコンストラクターが実行されますか?
実行結果は次のとおりです。
実行結果からわかるように、親クラスAは引き続きパラメーターなしの構築を実行します。
言い換えると、親クラスコンストラクターがサブクラスコンストラクターで明示的に指定されていない場合、親クラスのパラメーターなしの構築がデフォルトで実行されます[結論1]
言い換えると、親クラスコンストラクターがサブクラスコンストラクターで明示的に指定されている場合(つまり、super(n);が最初の行に追加されている場合)、親クラスのパラメーター化された構築が実行されます[結論2]
我们证明如下:(为了简化,我们注释掉静态成员,从而更好的专注于问题)
当我们在子类中显式地指定有参构造后(即在子类有参构造的第一行加上super(n);),则输出结果如下
public B(int n) {
//super();
super(n);
System.out.println("B的有参构造");
this.numB2 = n;
}
サブクラスのデフォルトのすべてのコンストラクター[デフォルト:親クラスのコンストラクターは明示的に指定されていません]は、親クラスの空のパラメーターを使用してコンストラクターにアクセスします[結論3]
证明如下:
我们注释掉父类的无参构造,但是保留子类的无参构造,结果发生编译错误
原因:所有的构造器包括子类中的无参构造,都会默认访问父类中的无参构造,这里子类中存在的无参构造器就没有显式指定父类构造器(子类中的有参构造器已经在第一行指定了父类构造器),但是父类中没有无参构造。
解决办法:我们可以注释掉子类中的无参构造,这样,子类中的所有构造器都显式地指定了父类构造器。