作者 | 小明菜市场

来源 | 小明菜市场(ID:fileGeek)

头图 |  CSDN 下载自东方IC

前言



从一道面试题说起

A:接口里可以写方法吗?

B:可以的,默认就是抽象方法。

A:那接口里可以写实现方法吗?

B:不可以,所有的方法必须是抽象的。

A:你确定?

B:确定。。。。

好吧。这的的确确让人有点怀疑,所以本文就从这开始着手,

这里介绍一个Java8的新特性,接口增强

静态方法和默认方法

我们可以在Comparator接口的源码中,看到大量类似下面这样的方法声明。

//default关键字修饰的默认方法
    default Comparator<T> thenComparingInt(ToIntFunction<? super T> keyExtractor) {
        return thenComparing(comparingInt(keyExtractor));
    }
    //Comparator接口中的静态方法
    public static <T extends Comparable<? super T>> Comparator<T> naturalOrder() {
        return (Comparator<T>) Comparators.NaturalOrderComparator.INSTANCE;
    }

其中thenComparingInt()就是一个默认方法,它使用 default 关键字修饰。这是Java8引入的新功能,接口中的可以声明默认方法和静态方法。

默认方法带来的多继承问题

在此之前,Java中的类只支持多重继承,不支持多继承,现在有了默认方法,你可以以另外一种方式实现类的多继承行为,即,一个类实现多个接口,而这几个接口都有声明自己的默认方法。

这里面引发了一个多继承问题,设想一下,假如一个类从多个接口中继承了它们声明的默认方法,而这几个默认方法使用的都是相同的函数签名,那么程序运行时,类会选择调用哪个方法。

第一份代码

@Test
    public void test2() {
        new C().hello();//result: hello from D
    }


    interface A {
        default void hello() {
            System.out.println("heelo from A");
        }
    }


    interface B extends A {
        default void hello() {
            System.out.println("heelo from B");
        }
    }


    class D implements A{
        public void hello() {
            System.out.println("hello from D");
        }
    }


    class C extends D implements A, B{
    }

这份代码输出的结果是 hello from D,可以看到 C 类的父类D,父接口A,父接口B都定义了一个相同函数签名的hello(),最后实际调用的是父类D中声明的方法。

第二份代码



 @Test
    public void test4() {
        new I().hello();//result: heelo from G
    }


    class I implements G, H { }
    
    interface G extends E {
        default void hello() {
            System.out.println("heelo from G");
        }
    }
    
    interface H extends E { }


    interface E {
        default void hello() {
            System.out.println("heelo from E");
        }
    }

代码清单二的输出结果是 hello from G,可以看到 I 类的父接口 G,父接口E都定义了一个相同函数签名 hello(),最后实际调用的是父类接口G中声明的方法。

第三份代码



 @Test
    public void test3() {
        new F().hello(); //result: heelo from E
    }


    interface A {
        default void hello() {
            System.out.println("heelo from A");
        }
    }


    interface E {
        default void hello() {
            System.out.println("heelo from E");
        }
    }


    class F implements A, E {
        public void hello() {
            //这里接口A和E不再具有继承关系,需显式的选择调用接口E或A中的方法,否则无法通过编译
            E.super.hello();
        }
    }

第三份代码,类F必须显式的覆盖父接口的hello方法,否则无法通过编译器的检测,因为编译器无法确定父接口A和父接口E中的默认方法哪一个优先。这种情况下,如果你想调用某个父接口的默认方法,可以使用 接口名.super.默认方法名这种方式进行调用。

总结

Java8的新特性:接口中可以声明默认方法和静态方法。另外,接口默认方法带来的多继承问题,即,如果一个类使用相同的函数签名从多个地方继承了方法,通过这三种规则可以判断:

  1. 类中的方法优先级最高。类或父类中声明的方法的优先级高于任何声明为默认方法的优先级。

  2. 如果无法依据第一条进行判断,那么子接口的优先级更高:函数签名相同时,优先选择有最具体实现的默认方法的接口,即如果B继承了A,那么B就比A更加具体。

  3. 最后, 如果还是无法判断, 继承了多个接口的类必须通过显式覆盖和调用期望的方法, 显式地选择使用哪一个默认方法的实现(调用语法:  接口名.super.默认方法名 )。

Logo

20年前,《新程序员》创刊时,我们的心愿是全面关注程序员成长,中国将拥有新一代世界级的程序员。20年后的今天,我们有了新的使命:助力中国IT技术人成长,成就一亿技术人!

更多推荐