问题描述
- 一个典型的问题,关于继承成员内部类的子类的构造函数
-
初学JAVA,没有C币,但是希望大家能够帮忙解决一下问题:
先上代码:package Test; public class test9 { public static void main(String[] args){ Outer outer=new Outer("test"); new Extender(outer,10); } } class Outer{ public Outer(String str){} class Inner{ public Inner(){} public Inner(int a){ System.out.println("Inner class has been created"); } } } class Extender extends Outer.Inner{ public Extender(Outer outer,int a){ //outer.super(a); //语句1 outer.new Inner(a); //语句2 } }
问题出在对Extender类的构造函数的定义上:
之前都一直习惯,直接在构造函数的首行显示引用超构造函数,但是也是一知半解。前几天突然发现这个问题:
我们都知道,super这个关键字其实就是代替了一段字符串“new 父类名”,所以我就想既然用“outer.super(a)”能够编译通过,那“outer.new Inner(a)”应该也能编译通过,于是我做了以下几个实验:
1、执行语句1,不执行语句2:编译通过,打印出“Inner class has been created"
2、执行语句1,也执行语句2:编译通过,打印出两行“Inner class has been created”
3、不执行语句1,执行语句2:编译不通过,报错:“由于某些中间构造函数调用,没有任何类型 Outer 的外层实例可用”我想不明白的就是,既然实验2中,new outer.Inner(a)能够创建对象成功,为什么在实验3里面,单独运行时就报错呢???
解决方案
java内部类的构造器必须连接到指向其外围类对象的引用,也就是构造内部类必须给它一个外部类对象的引用,内部类依赖于外部类对象,因此在继承内部类的时候,需要在继承类中的构造器中手动加入对基类(外围类)构造器的调用。因为,在继承类实例化时,并不存在一个外围类对象,以让继承类的实例去连接到外围类。所以,我们需要创建一个外围类,然后用一个特定的语法来表明内部类与外围类的关系.
题主上文中的第三个实验并没有调用外围类的构造器,而只是建立了一个内部类实例而已,这个和外围类的构造是没有关系的
解决方案二:
首先我也是刚刚接触java,看你这个题也挺有意思,也来凑凑热闹
No enclosing instance of type Outer is available due to some intermediate constructor invocation
提示的错误我用有道翻译了下大概意思是这样的:由于某些构造函数的调用,没有Outer内部的实例;
我的理解是outer.new Inner(a);这句只是创建了一个实例,并非调用父类的构造函数,我做了一下试验:
1 。outer.super(a); //语句1
2 。 Outer.Inner in=outer.new Inner(a);//语句2
将语句1.2全注释了,结果提示错误跟上面一样。然后又将语句2加上父类的引用是可以的,也就印证了我的想法,语句2只是创建了一个实例,而非初始父类,要初始化父类构造函数必须用super;课上老师也说到过隐式的调用是在你没有写构造函数时默认给一个super(),这里我也试过将Inner里的构造都注释调不过依然会报同样的错误,所以我又把Outer里的构造也一起都注释掉了,改成继承为Outer结果语句2就可以单独存在了。综上所述:我的理解是隐式的构造传递只会给你传一个super(),而不会是out.super();而你的语句2充其量只是一个实例。这个问题应该是编译器的问题吧?所以我也找不到答案,希望大神能过解惑!