转自鸟窝 博主写的挺详细,不了解的看一看啊

以前经常谈论的Java对比c++的一个优势是Java中没有多继承的问题。 因为Java中子类只能继承(extends)单个父类, 尽管可以实现(implements)多个接口,但是接口中只有抽象方法,方法体是空的,没有具体的方法实现,不会有方法冲突的问题。

这些都是久远的说法了,自从今年Java 8发布后, 接口中也可以定义方法了(default method)。 之所以打破以前的设计在接口中
增加具体的方法, 是为了既有的成千上万的Java类库的类增加新的功能, 且不必对这些类重新进行设计。 比如, 只需在Collection接口中
增加default Stream<E> stream(), 相应的SetList接口以及它们的子类都包含此的方法, 不必为每个子类都重新copy这个方法。

这是一个折衷的设计,带来的问题就是为Java引入了多继承的问题。 我们知道, 接口可以继承接口, 类可以继承类和实现接口。 一旦继承的类和实现的接口中有
相同签名的方法, 会出现什么样的状况呢? 本文将探讨各种情况的多继承, 以便能清楚的理解Java多继承的规则。

接口继承多个父接口

假定有三个接口Interface A, Interface B, Interface C, 继承关系如下:

1
2
3
4
5
6
7
8
9
+---------------+ +------------+
| Interface A | |Interface B |
+-----------^---+ +---^--------+
| |
| |
| |
+-+------------+--+
| Interface C|
+------------+

A,B拥有相同签名的默认方法default String say(String name), 如果接口C没有override这个方法, 则编译出错。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
interface A {
default String say(String name) {
return "hello " + name;
}
}
 
interface B {
default String say(String name) {
return "hi " + name;
}
}
 
interface C extends A,B{
 
}

错误信息:

1
2
3
4
5
6
C:\Lambda\src>javac -J-Duser.country=US com\colobu\lambda\chap
ter3\MultipleInheritance1.java
com\colobu\lambda\chapter3\MultipleInheritance1.java:17: error: interface C inherits unrelated defaults for say(String) from types A and B
static interface C extends A,B{
^
1 error

我们可以在子接口C中覆盖override这个方法, 这样编译就不会出错了:

1
2
3
4
5
interface C extends A,B{
default String say(String name) {
return "greet " + name;
}
}

注意方法签名不包括方法的返回值, 也就是仅仅返回值不同的两个方法的签名也是相同的。
下面的代码编译不会出错,因为AB的默认方法不同, C隐式继承了两个默认方法。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
interface A {
default void say(int name) {
 
}
}
 
interface B {
default void say(String name) {
 
}
}
 
interface C extends A,B{
 
}

但是有的情况下即使是不同签名的方法也是很难分辨的:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
interface A {
default void say(int a) {
System.out.println("A");
}
}
 
interface B {
default void say(short a) {
System.out.println("B");
}
}
 
interface C extends A,B{
 
}
 
static class D implements C {
 
}
 
public static void main(String[] args) {
D d = new D();
byte a = 1;
d.say(a); //B
}

Java会选择最适合的方法, 请参看Java规范 15.12.2.5

接口多层继承

下面看一下多层继承的问题。 继承关系如下图, A2继承A1, C继承A2。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
+---------------+
| Interface A1 |
+--------+------+
|
|
|
+--------+------+
| Interface A2 |
+-------+-------+
|
|
|
+-------+--------+
| Interface C |
+----------------+

基于我们以前对类继承的认识, 很容易知道C会继承A2的默认方法,包括直接定义的默认方法, 覆盖的默认方法,以及隐式继承于A1接口的默认方法。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
interface A {
default void say(int a) {
System.out.println("A");
}
 
default void run() {
System.out.println("A.run");
}
}
 
interface B extends A{
default void say(int a) {
System.out.println("B");
}
 
default void play() {
System.out.println("B.play");
}
}
interface C extends A,B{
 
}

多层多继承

上面一个例子还是单继承的例子, 如果如下图的多继承呢?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
+---------------+
| Interface A1 |
+--------+------+
|
|
|
+--------+------+ +---------------+
| Interface A2 | | Interface B |
+-------+-------+ +---------+-----+
| +---------+---------^
| |
| |
+-------+-------++
| Interface C |
+----------------+

如果A2和B拥有相同签名的方法,这和第一个例子一样。 如果不想编译出错,可以覆盖父接口的默认方法,还可以调用指定父接口的默认方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
interface A1 {
default void say(int a) {
System.out.println("A1");
}
}
 
interface A2 extends A1 {
 
}
 
interface B {
default void say(int a) {
System.out.println("B");
}
}
 
interface C extends A2,B{
default void say(int a) {
B.super.say(a);
}
}

更复杂的多层多继承

1
2
3
4
5
6
7
8
9
10
11
12
13
+--------------+
| Interface A1 |
+------+------++
| ^+-------+
| |
+-------+-------+ |
| Interface A2 | |
+------------+--+ |
^--++ |
| |
+--+------+-----+
| Interface C |
+---------------+

接口A2继承A1, 接口C继承A2和A1。 代码如下,

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
interface A1 {
default void say() {
System.out.println("A1");
}
}
 
interface A2 extends A1 {
default void say() {
System.out.println("A2");
}
}
 
 
 
interface C extends A2,A1{
 
}
 
static class D implements C {
 
}
 
public static void main(String[] args) {
D d = new D();
d.say();
}

以上代码不会编译出错,运行输出A2
可以看到接口C会隐式继承子接口的方法, 也就是子接口A2的默认方法。

类继承

如果继承关系类型全部是类, 那么由于类依然是单继承的, 不会有多继承的问题。

类和接口混杂

我们把第一个例子中的其中一个接口换成类,会出现什么现象呢。

1
2
3
4
5
6
7
8
+-------------+ +-----------+
| Interface A | | Class B |
+-----------+-+ +-----+-----+
^-+ +--+-----^
| |
+---+----+-+
| Class C |
+----------+

以下代码不会编译出错:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
interface A {
default void say() {
System.out.println("A");
}
}
 
static class B {
public void say() {
System.out.println("B");
}
}
 
 
 
static class C extends B implements A{
 
}
 
 
public static void main(String[] args) {
C c = new C();
c.say(); //B
}

结果输出B
可以看出, 子类优先继承父类的方法, 如果父类没有相同签名的方法,才继承接口的默认方法。

结论

关于java8 interface的default方法的更多相关文章

  1. Java8中的default方法

    default方法 Java 8中引入了一个新的概念,叫做default方法,也可以称为Defender方法,或者虚拟扩展方法(Virtual extension methods). Default方 ...

  2. java8新特性:interface中的static方法和default方法

    java8中接口有两个新特性,一个是静态方法,一个是默认方法. static方法 java8中为接口新增了一项功能:定义一个或者多个静态方法. 定义用法和普通的static方法一样: public i ...

  3. Java8新特性interface中的static方法和default方法

    static方法 java8中为接口新增了一项功能:定义一个或者更多个静态方法.用法和普通的static方法一样. 代码示例 public interface InterfaceA { /** * 静 ...

  4. Java8新特性(一)_interface中的static方法和default方法

    什么要单独写个Java8新特性,一个原因是我目前所在的公司用的是jdk8,并且框架中用了大量的Java8的新特性,如上篇文章写到的stream方法进行过滤map集合.stream方法就是接口Colle ...

  5. java8中接口中的default方法

    在java8以后,接口中可以添加使用default或者static修饰的方法,在这里我们只讨论default方法,default修饰方法只能在接口中使用,在接口种被default标记的方法为普通方法, ...

  6. 深入学习Java8 Lambda (default method, lambda, function reference, java.util.function 包)

    Java 8 Lambda .MethodReference.function包 多年前,学校讲述C#时,就已经知道有Lambda,也惊喜于它的方便,将函数式编程方式和面向对象式编程基于一身.此外在使 ...

  7. Java8 新特性 默认方法

    默认方法为什么出现 默认方法的出现是因为在java8设计的过程中,因为加入了Lamdba表达式,和函数式接口,所以在非常多的接口里面要加入新的方法,但是如果在接口里面直接加入新的方法,那么以前写的所有 ...

  8. java8中接口default、static新特性,与抽象类区别

    之前Java接口中的方法默认都是public abstract,成员变量默认都是public static final,偶然发现接口中可以有default类型的方法,才知道java8中接口可以有自己的 ...

  9. Java8中的默认方法

    作者:汤圆 个人博客:javalover.cc 前言 大家好啊,我是汤圆,今天给大家带来的是<Java8中的默认方法>,希望对大家有帮助,谢谢 文章纯属原创,个人总结难免有差错,如果有,麻 ...

随机推荐

  1. 检测Java程序运行时间的2种方法(高精度的时间[纳秒]与低精度的时间[毫秒])

    第一种是以毫秒为单位计算的. 代码如下: long startTime=System.currentTimeMillis(); //获取开始时间 doSomeThing(); //测试的代码段 lon ...

  2. C# 获取本机CPU序列号,MAC地址,硬盘ID,本机IP地址,计算机名,物理内存,PC类型

    首先引入服务 然后 调用 本文转载自http://blog.sina.com.cn/s/blog_7eeb43210101hf7f.html public class Computer { publi ...

  3. Function类型

    1.每个函数都是Function类型的,和其他引用类型一样都具有属性和方法.函数也是对象,因此函数实际上是一个指向函数对象的指针. 函数声明语法定义: 方法1: function sum(num1,n ...

  4. crc循环冗余校验

    循环冗余校验(Cyclic Redundancy Check, CRC)是一种根据网络数据包或电脑文件等数据产生简短固定位数校验码的一种散列函数,主要用来检测或校验数据传输或者保存后可能出现的错误.它 ...

  5. JavaBean的toString方法工具类

    import java.lang.reflect.Field;import java.lang.reflect.InvocationTargetException;import java.lang.r ...

  6. sql 将8位字符串转换成日期型

    将8位字符串转换成日期型,方法如下: ),)

  7. 实用的PHP功能详解(一)_php glob()用法

    一.使用glob()查找文件 大部分PHP函数的函数名从字面上都可以理解其用途,但是当你看到 glob() 的时候,你也许并不知道这是用来做什么的,其实glob()和scandir() 一样,可以用来 ...

  8. HTML5.dcloud.io-stream-app

    dcloud.io提出的Stream App 本文仅仅是关于dcloud.io提出的SteamApp初探,所有内容请参考其官网. 1. Application promotion by scaning ...

  9. VBA学习之关于数据透视表的应用

    工作中很多地方需要同时处理多个数据表,而且用数据透视表进行排版,排序,计算字段,一个一个的做非常累,这里给出批量处理的方法. 学习VBA之前最好懂一点点VB的基础知识,因为里面的很多语法问题都是由VB ...

  10. XAF学习笔记1

    写软件很多年了.看过的框架用过的框架非常多.一直想要一种框架,说不出的心烦重复的数据库设计,重复的Model生成,重复的界面设计 一直寻寻觅觅,终于找到一个框架,DEV的XAF,DEV控件用了N年了, ...