2 xy83138 xy83138 于 2016.04.12 08:12 提问

一个典型的问题,关于继承成员内部类的子类的构造函数

初学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里面,单独运行时就报错呢???

2个回答

chen956
chen956   2016.04.12 09:22
已采纳

java内部类的构造器必须连接到指向其外围类对象的引用,也就是构造内部类必须给它一个外部类对象的引用,内部类依赖于外部类对象,因此在继承内部类的时候,需要在继承类中的构造器中手动加入对基类(外围类)构造器的调用。因为,在继承类实例化时,并不存在一个外围类对象,以让继承类的实例去连接到外围类。所以,我们需要创建一个外围类,然后用一个特定的语法来表明内部类与外围类的关系.
题主上文中的第三个实验并没有调用外围类的构造器,而只是建立了一个内部类实例而已,这个和外围类的构造是没有关系的

chen956
chen956 虽然在继承类中写了outer.new Inner(a),但是这句话调用的内部构造并不是继承类创建实例时需要调用的存在,必须给他一内部类构造。这个是继承类初始化的原理。之前说的外围类初始化说错了。这个程序唯一和外围类有关系就是内部类属于外围类,内部类在使用时不能单独存在,必须给它一个外部类对象的引用。
一年多之前 回复
chen956
chen956 说错了,是这样:outer.new Inner(a)是新建一个内部类实例,而outer.super(a)是在继承类初始化的时候可以初始化内部类,而不是新建一个内部类实例。还是之前的回答:就像普通的继承一样子类需要调用super()方法初始化基类。这里就可以表述为继承类继承内部类,需要调用super来初始化内部类(基类),但是内部类是属于外围类的,因此需要一个外围类的引用来初始化内部基类。也就是outer.super().
一年多之前 回复
chen956
chen956 这个就像最初回答你的:内部类依赖于外部类对象,因此在继承内部类的时候,需要在继承类中的构造器中手动加入对基类(外围类)构造器的调用。即用一个特定的语法来表明内部类与外围类的关系
一年多之前 回复
xy83138
xy83138 回复清泉流: 您好!我的理解是:outer是外部类的一个实例了,所以再想创建一个内部类的对象的时候,不需要再去初始化一个外部类的对象了,直接把想要创建的那个内部类的对象连接到outer上,利用"outer.new Inner(a)"中的"outer."部分来完成,这样不行吗???
一年多之前 回复
chen956
chen956 new Inner怎么会满足其外围类的初始化(inner只是内部类,不是子类)。outer只是一个引用,需要靠这个引用完成外围类的初始化
一年多之前 回复
xy83138
xy83138 回复清泉流: 真的十分谢谢您的回答,我们这些自学者真的很需要像您这样的人帮忙!谢谢!!!关于"outer.new Inner(a)",我还有几个疑问:第一,“执行子类的构造函数之前,必须要首先显示或者隐式地调用父类的构造函数”,”outer.new Inner(a)“里的”new Inner(a)“部分就满足了这部分要求;第二,”创建内部类的实例时,必须让该实例连接到一个外部类实例“,"outer.new Inner(a)"里的"outer."部分,就满足了这个要求。。为什么这两个要求都满足了,还是不行啊??
一年多之前 回复
chen956
chen956 传入外部对象的引用是为了调用外围类的构造器,就像普通的继承一样子类需要调用super()方法初始化基类,只不过内部类的继承需要外围类的引用才能使得继承类可以调用基类(外围类)的super().
一年多之前 回复
xy83138
xy83138 谢谢您的回答,可是我还是有一点不明白。您说必须要有一个外部类对象的引用,继承类的传入参数里不是已经传了一个外部类的对象进去了吗??还是不行??
一年多之前 回复
xy83138
xy83138 谢谢您的回答,可是我还是有一点不明白。您说必须要有一个外部类对象的引用,继承类的传入参数里不是已经传了一个外部类的对象进去了吗??还是不行??
一年多之前 回复
xy83138
xy83138 谢谢您的回答,可是我还是有一点不明白。您说必须要有一个外部类对象的引用,继承类的传入参数里不是已经传了一个外部类的对象进去了吗??还是不行??
一年多之前 回复
qq_34108945
qq_34108945   2016.04.13 00:40

首先我也是刚刚接触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充其量只是一个实例。这个问题应该是编译器的问题吧?所以我也找不到答案,希望大神能过解惑!

qq_34108945
qq_34108945 回复xy83138: 试不出结果,等其他人来解答吧!
一年多之前 回复
xy83138
xy83138 谢谢参与讨论!!大家一起进步!!有一点疑问:语句2中的new Inner(a),都使用new关键字了,还算没有调用父类的构造函数??
一年多之前 回复
Csdn user default icon
上传中...
上传图片
插入图片
准确详细的回答,更有利于被提问者采纳,从而获得C币。复制、灌水、广告等回答会被删除,是时候展现真正的技术了!