前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java8中的默认方法

Java8中的默认方法

原创
作者头像
汤圆学Java
修改2021-04-14 14:20:58
5540
修改2021-04-14 14:20:58
举报
文章被收录于专栏:汤圆学Java汤圆学Java

作者:汤圆

个人博客:javalover.cc

前言

大家好啊,我是汤圆,今天给大家带来的是《Java8中的默认方法》,希望对大家有帮助,谢谢

文章纯属原创,个人总结难免有差错,如果有,麻烦在评论区回复或后台私信,谢啦

简介

在Java之前,我们接触到的接口,都是只定义方法,不实现方法

(你看下面这几个人,像不像接口)

指手画脚
指手画脚

但是到了Java8就不一样了,因为在接口中新增了默认方法

这样的话,有些活,就可以交给接口自己去做了,而不用实现类去做(Java你这是在收买人心啊)

我们下面以问答的形式来介绍默认方法的相关知识点(据说问答模式可以让人更好地记忆?)

目录

  • 什么是默认方法?
  • 为啥要提供默认实现?
  • 那我如果不提供呢?
  • 这个功能主要是针对谁?
  • 实现了默认方法的接口和抽象类有区别吗?
  • 是不是可以说实现了多重继承?

正文

什么是默认方法

默认方法是接口中用default修饰的方法,其中包含方法内容

比如下面这个:

代码语言:javascript
复制
public interface InterfaceDemo {
    // 普通方法,只定义,不实现
    void oldFun();
    // 默认方法,又定义,又实现
    default void newFun(){
        System.out.println("newFun");
    }
}

为啥要提供默认方法呢?

为了向后兼容(这也是导致Java变得臃肿的原因之一)。

因为升级系统时,难免会有一些新功能需要加入,此时如果接口类新增了方法,那么实现类就必须同步修改实现;

这样工作量还是很大的,而且很容易出错。

所以Java8开始,推出了接口的默认方法这个功能,使得接口升级变得更加平滑

比如下面的代码:InterfaceDemo就是上面那个接口

代码语言:javascript
复制
public class UserDemo implements InterfaceDemo{
    @Override
    public void oldFun() {
        System.out.println("oldFun");
    }
​
    public static void main(String[] args) {
        UserDemo demo = new UserDemo();
        /**
         *  InterfaceDemo升级后,新增了newFun方法
         *  但是由于newFun是默认方法,有提供实现内容
         *  所以这里的子类 UserDemo就可以直接使用
          */
        demo.newFun();
    }
}

我们可以看到,UserDemo没有实现新的方法newFun(),但是也可以编译运行,并直接调用newFun()

这就是默认方法的好处:对实现类来说是无痛升级的

如果不提供呢?

不提供的话,接口类升级时,系统有两个选择

  1. 实现类升级:
    • 实现类老老实实地按照接口升级后的方法,进行同步修改实现,但是工作量大
  2. 实现类不升级:
    • 实现类不升级也是可以的,只要不引入接口类的新版本就可以了,那么这个时候系统还是可以运行的,这没啥问题。但是谁能保证一辈子都不更新系统呢?如果更新系统时,接口类库升级到新版本,那么编译还是通不过

主要针对谁?

接口的默认方法主要是针对类库设计者

实现了默认方法的接口和抽象类有区别吗

区别没有之前那么多,但还是有的:

  1. 抽象类单继承,接口类多实现
  2. 抽象类中的属性定义时不需要初始化,接口类的属性定义时要初始化(默认修饰符为public static final)

是不是可以说Java现在也实现了多重继承?

可以这么说。

但是现在面临的一个新问题,就是多重继承带来的二义性问题,有点类似之前介绍的致命方块(也叫菱形问题)

如下面的UML图所示

二义性
二义性

比如上面这种,你无法知道A会调用哪个接口的fun方法

所以编译器会报错:

代码语言:javascript
复制
com.jalon.java8.defaultmethod.A inherits unrelated defaults for fun() from types com.jalon.java8.defaultmethod.B and com.jalon.java8.defaultmethod.C

解决办法:

  • 先覆写fun方法
  • 再显示声明调用哪个接口的fun方法

代码如下:

代码语言:javascript
复制
​
public class A implements B,C{
​
    @Override
    public void fun(){
        // 显示调用B的默认方法
        B.super.fun();
    }
​
    public static void main(String[] args) {
        A a = new A();
        // 这里会打印B的fun
        a.fun();
    }
}
interface D{
    default void fun(){
        System.out.println("D");
    }
}
interface B extends D{
    @Override
    default void fun(){
        System.out.println("B");
    }
}
interface C extends D{
    @Override
    default void fun(){
        System.out.println("C");
    }
}

总结

  • 什么是默认方法:接口中用default修饰且包含方法内容的方法
  • 为什么要提供默认方法:向后兼容,使系统平滑过渡;主要针对类库设计者
  • 多重继承带来的问题:二义性,也叫菱形问题;解决办法就是子类尽量覆写默认方法并显式声明调用哪个方法(实际上这个问题很少出现,因为它属于编译错误,写代码时随时可以发现)

后记

最后,感谢大家的观看,谢谢

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言
  • 简介
  • 目录
  • 正文
    • 什么是默认方法
      • 为啥要提供默认方法呢?
        • 如果不提供呢?
          • 主要针对谁?
            • 实现了默认方法的接口和抽象类有区别吗
              • 是不是可以说Java现在也实现了多重继承?
              • 总结
              • 后记
              领券
              问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档