Java之继承和实现

用过Java的人都知道,Java有一个特点就是单继承、多实现(这一点跟C++不一样,C++是可以多继承的),那么为什么会是这样子的呢?让我们来一探究竟!
老规矩,打开IDE工具,我们还是用例子来试试看:
首先我们定义一个最简单的类B,如下:

public class B {
    private String name;
    public void setName(String name) {
        this.name = name;
    }
   public String getName() {
        return name;
    }
}

然后再定义一个类A,并让它继承B,如下:

public class A extends B{
    @Override
    public void setName(String name) {
        super.setName(name);
    }
}

注意到上面的 super.setName(name);这句话的意思是调用父类中的相应方法。
那么,问题就来了,比如允许多继承的话,如果多个父类中有相同的方法,那么子类该super哪个父类的方法呢?当然,还有属性也是一样的,所以,为了避免歧义及纠结不清,Java的类采用了单继承。
如果你强行在B后面加上一个逗号并再加上另一个类的话,IDE工具会给出注意的错误提醒:“Class cannot extend multiple classes”。注意到这个提醒的主语是“Class”,意思就是:类不能继承多个类,那么,言下之意,还是存在可以多继承的,比如说接口,下面我们继续举例:

public interface C{
     String name="二狗子";
     void setName();
     void setName(int i);
}
public interface D {
    void setName();
    void setName(String name);
}
public interface E extends C,D{
    void setName(long l);
}

以上,我们定义了3个接口,其中E继承于C、D,这已经说明在接口上完全是可以多继承的!
我们先来注意一下上面接口C中的一个属性:name,按理来说,这个属性前面没有加任何作用域说明(public,protected,private),那么这个属性应该属于default(或叫做package)作用域,起码在类中我们是这么规定作用域的,然而在接口中却不是如此!
在接口中,所有属性都是 static final修饰的,即是常量,所以,比如上面的例子,我们可以通过C.name直接访问到该属性。
再回到接口多继承的问题,为什么接口允许多继承呢?其实这也是接口的特性造成的,因为接口规定其不能有方法体,你不信可以试试在接口中的方法加上一对大括号,然后IDE工具就会给出这样的错误提醒:“ Interface methods cannot have body”。所以多继承并不存在歧义的问题,就算有相同的方法也没事,反正最终实现类都会重写接口中的方法。
如上面的例子,因为E继承于C、D,那么E.name会获取到“二狗子”这个常量,那么问题又来了,如果我在D中又定义了一个常量: String name="二蛋子"; 那么,E.name会是“二狗子”还是"二蛋子"呢?
其实,这会压根就通不过,IDE工具给出这样的错误提醒:“Reference to 'name' is ambiguous, both 'C.name' and 'D.name' match” ,这样才是正常的,毕竟正如上面所说的,接口中所有属性都是 static final修饰的,那么,其值应该在编译时期就确定下来了,像上面的例子,接口多继承,存在了E.name存在两个不同的常量值,这在编译期就通不过了。

我们再回到类的多实现上,借用上面的C、D接口,继续举个例子:

public class A extends B implements C,D {
    @Override
    public void setName() {
    }

    @Override
    public void setName(int i) {
    }
}

如上面例子,类A继承了类B并实现了接口C、D,上面的两个方法:setName() 和setName(int i) 是通过IDE工具帮我们生成的,也就是说这两个方法是必须要实现的方法!
那么问题来了,我们观察一下C接口,有setName()和setName(int i)两个抽象方法,而D接口有setName()和setName(String name)两个抽象方法,其中setName()是相同的方法,只需实现一次,这没什么好说的,那么,为什么不要求实现setName(String name)这个抽象方法呢?
其实原因也很简单,那就是因为类A继承了类B,而类B中有setName(String name)这个方法,接口D中的setName(String name)抽象方法由A的父类B代为实现了,如何证明这一点呢,我们再看如下的测试:

public class A extends B implements C,D {
    @Override
    public void setName() {
        MyTest myTest=new MyTest(this);//因为A实现了D接口,这里直接用this即可
        D d = myTest.getD();
        d.setName("二蛋子");//调用D接口的setName(String name)
    }
    @Override
    public void setName(int i) {

    }
     class MyTest{
        private D d;
        MyTest(D d){
            this.d=d;
        }
        D getD() {
            return d;
        }
    }
    public static void main(String[] args) {
        A a=new A();
        System.out.println("A继承B,B中的name:"+a.getName());
        a.setName();
        System.out.println("A继承B,B中的name:"+a.getName());
    }
}

结果如下:

A继承B,B中的name:null
A继承B,B中的name:二蛋子

上面这个测试也很简单,实践证明,调用D接口的setName(String name),会使得继承的父类B中的name发生变化,以此证明了D接口的setName(String name)是由父类B代为实现了。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 专业考题类型管理运行工作负责人一般作业考题内容选项A选项B选项C选项D选项E选项F正确答案 变电单选GYSZ本规程...
    小白兔去钓鱼阅读 9,059评论 0 13
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,148评论 1 32
  • 小编费力收集:给你想要的面试集合 1.C++或Java中的异常处理机制的简单原理和应用。 当JAVA程序违反了JA...
    八爷君阅读 4,681评论 1 114
  • 几天前,收到一张从鼓浪屿寄来的明信片,画面是一片海。 打电话给她,听她说着自己在鼓浪屿的日子过得很开心,早上迎着日...
    梓瞳阅读 675评论 1 1
  • 农历的十二月十六这天是个好日子。天还没亮,“噼里啪啦”的声响开始在徐家村的上空热闹起来,鞭炮声和礼花撕开了黑暗的夜...
    大野泽的风阅读 719评论 2 2