JAVA 8 默认方法-Default Methods

什么是默认方法-Default Methods?简单的说,就是可以在接口中定义一个已实现方法,且该接口的实现类不需要实现该方法.如下示例:

interface GreetingService  {
        void sayMessage(String message);
        
        //可以在接口中定义默认方法
        default void sayHello(){
            System.out.println("Hello");
        }
    }
    
    //实现类不需要实现接口中的默认方法
    class GreetingServiceImpl implements GreetingService{
        @Override
        public void sayMessage(String message)
        {

        }
    }
1、为什么要有默认方法?

主要是为了方便扩展已有接口;如果没有默认方法,加入给JDK中的某个接口添加一个新的抽象方法,那么所有实现了该接口的类都得修改,影响将非常大。

使用默认方法,可以给已有接口添加新方法,而不用修改该接口的实现类。当然,接口中新添加的默认方法,所有实现类也会继承该方法。

举个例子,在Java 8的Iterable接口中,新增了一个默认方法forEach,也正因为forEach是默认方法,才不用修改所有Iterable接口的实现类。

Iterable接口新增的forEach方法如下(入参是一个函数式接口,因此支持Lambda表达式):

default void forEach(Consumer<? super T> action) {
        Objects.requireNonNull(action);
        for (T t : this) {
            action.accept(t);
        }
}

因为Collection接口继承了Iterable接口,所以我们可以在集合类中使用forEach方法,如下,这里使用了方法引用(一种更加紧凑的Lambda表达式)

List<String> list = new ArrayList<String>();
        list.add("001");
        list.add("002");
        list.forEach(System.out::println);
可见,我们在未破坏Iterable接口实现类的前提下,给Iterable接口的所有实现类添加了一个新方法forEach,这在Java 8之前是不可能的。

2、重写Override默认方法
如果子类没有重写父接口默认方法的话,会直接继承父接口默认方法的实现;
如果子类重写父接口默认方法为普通方法,则与普通方法的重写类似;
如果子类(接口或抽象类)重写父接口默认方法为抽象方法,那么所有子类的子类需要实现该方法;


3、关于默认方法调用冲突

因为一个类是可以实现多个接口的,如果多个接口定义了同样的默认方法,那么子类如何调用父类的默认方法呢?具体调用流程如下:

1)首先,如果子类覆盖了父类的默认方法,那么什么也不用想,直接使用调用子类覆盖后的方法;

2)其次,优先选择调用更加具体的接口默认方法,什么意思呢,举个例子,如果A1接口继承A接口,那么A1接口相对A接口就更加具体,当C类实现了A1接口的时候,就优先调用A1接口的默认方法;

3)最后,如果C类同时实现A1接口和A2接口,且A1和A2有同名的默认方法,那么选择哪个接口的默认方法呢?答案是编译器报错,提示定义了重名的方法,快速修复方式是覆盖其中的一个即可;

关于这块内容,在网上看到一段有意思的代码,如下,知道为什么会报错吗?如果删除InterfaceB中的foo方法,是否还会报错?往InterfaceC中添加foo方法又会怎样?

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值