传统上,Java程序的接口是将相关方法按照预定组合到一起的方式。实现接口的类必须为接口中定义的方法提供一个实现,或者从父类中集成它的实现。但是,一旦类库的设计者需要更新接口,向接口中加入新的方法时候,这种方式就会出现问题。现存的类为了适应新的接口约定也要进行修改。

  Java 8为了解决这个问题,现在接口支持在生命方法的同时提供实现,Java 8 允许在接口内声明静态方法,Java 8 允许在接口中创建默认方法。默认方法可以指定所有实现类的默认实现。这种模式可以让你平滑的进行接口的优化。

List中的sort方法就是一个静态方法:

    default void sort(Comparator<? super E> c) {
Object[] a = this.toArray();
Arrays.sort(a, (Comparator) c);
ListIterator<E> i = this.listIterator();
for (Object e : a) {
i.next();
i.set((E) e);
}
}

通过default关键字来在接口中定义默认方法。可以直接调用sort,对列表进行排序:

Arrays.asList(1,2,3,5,54).sort(Comparator.naturalOrder());

Comparator.naturalOrder是一个静态方法,它返回了一个Comparator对象,按照自然序列对其中的元素进行排序

    public static <T extends Comparable<? super T>> Comparator<T> naturalOrder() {
return (Comparator<T>) Comparators.NaturalOrderComparator.INSTANCE;
}

Collection中的Stream方法也是一个静态方法:

    default Stream<E> stream() {
return StreamSupport.stream(spliterator(), false);
}

Stream是通过StreamSupport调用了一个stream方法,来返回的Stream对象。并且,现在你知道spliterator 是怎么来的了吧,它也是Collection接口的一个默认方法。

默认方法的引用就是为了解决Java API这样的类库演进问题。因为向接口中增加方法会导致所有实现类受害。它们也都要进行更新。默认方法可以让所有实现类自动继承接口的一个默认实现。

定义默认方法

  使用default前缀,并加入方法体即可,比如定义一个sized接口:

public interface Sized {
int size();
default boolean isEmpty(){
return size() == 0;
}
}

所有的实现类,都会自动继承isEmpty的实现。

Java8中的抽象类和抽象接口

  抽象接口是指函数式接口中 包含的方法->如 Predicate中的test。

    一个类只能继承一个抽象类,但是一个类可以实现多个接口。

  一个抽象类可以设置实例变量,接口不能有实例变量。

可选方法

  有时,一个类实现了接口,不过却刻意的将一些方法的实现留白。以Iterator接口为例,Iterator接口定义了hasNext、next还有remove方法。Java 8 之前,由于用户通常不会使用该方法,remove方法常被忽略。因此,实现了Iterator接口的类通常会为remove方法放置一个空的实现,这些都是毫无用处的模板代码。

  采用默认方法之后,你可以为这种类型的方法提供一个默认的实现,这样实体类就无需在自己的实现中显示的提供一个空方法。

interface Iterator<T> { 

    boolean hasNext();
T next();
default void remove() {
throw new UnsupportedOperationException();
}
}

通过这种方式,可以减少很多无效的模板代码。无需再实现remove方法。

行为的多继承

  默认方法之前是无法优雅的实现多继承的,让一个类从多个来源重用代码的能力。因为java的类只能继承单一的类,但是一个类可以实现多个接口。比如ArrayList类的定义:

public class ArrayList<E> extends AbstractList<E>
implements List<E>, RandomAccess, Cloneable, java.io.Serializable

它继承了一个类,实现了4个接口。因此ArrayList实际上是7个类型的直接子类。在某种程度上,我们早就有了多继承。

由于Java 8中的接口也可以提供实现,类可以从多个接口中继承他们的行为。

比如你要实现一个接口,有的类需要可以放大缩小,不需要改变形状。有的类需要改变形状 不需要放大缩小。这时可以将 放大缩小放在一个接口中提供一个实现,放大缩小放在一个接口中,然后在需要的类中 各自实现各自的接口。继承各自的默认实现。不要都放在一个接口中。

解决冲突的规则

  我们知道,一个类只可以继承一个类,但是可以实现多个接口。随着Java 8 带来的默认方法的加入,有可能会出现类或者多个接口中的默认方法出现重复的问题。在这种情况下,

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

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

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

public class C implements B,A{
void hello(){
//显示指定哪个接口中的同名方法
B.supper.hello();
}
}

小结

  1.Java 8中的接口可以通过默认方法和静态方法提供代码的实现。

  2.默认方法的开头以关键字default修饰,方法体与常规类的方法相同。

  3.向发布的接口添加抽象方法不是源码兼容的。

  4.默认方法的出现能帮助库的设计者以后向兼容的方式演进API。

  5.默认方法可以用于创建可选方法和行为的多继承。

  6.当函数名重复时,可以按照优先级进行判断,类和父类实现最高,其他无法确定时,可以显示指定调用哪个。

Java 8 (8) 默认方法的更多相关文章

  1. Java 8 特性 —— 默认方法和静态方法

    Java 8 新增了接口的默认方法.简单说,默认方法就是接口可以有实现方法,而且不需要实现类去实现其方法.我们只需在方法名前面加个 default 关键字即可实现默认方法. 为什么要有这个特性?之前的 ...

  2. java 8 接口默认方法

    解决问题:在java8 之前的版本,在修改已有的接口的时候,需要修改实现该接口的实现类. 作用:解决接口的修改与现有的实现不兼容的问题.在不影响原有实现类的结构下修改新的功能方法 案例: 首先定义一个 ...

  3. java8-新特性--(接口的默认方法与静态方法)

    Java 8用默认方法与静态方法这两个新概念来扩展接口的声明. public interface Inte{ void method(); default void defaultMethod(){ ...

  4. Java8 新特性 默认方法

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

  5. 深入理解Java 8 Lambda(语言篇——lambda,方法引用,目标类型和默认方法)

    作者:Lucida 微博:@peng_gong 豆瓣:@figure9 原文链接:http://zh.lucida.me/blog/java-8-lambdas-insideout-language- ...

  6. Java 8函数编程轻松入门(三)默认方法详解(default function)

    default出现的原因 Java 8中对API最大的改变在于集合类,Java在持续演进,但是它一直保持着向后兼容. 在Java 8中为Collection接口增加了stream方法,这意味着所有实现 ...

  7. JAVA 8 默认方法-Default Methods

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

  8. [转]深入理解Java 8 Lambda(语言篇——lambda,方法引用,目标类型和默认方法)

    以下内容转自: 作者:Lucida 微博:@peng_gong 豆瓣:@figure9 原文链接:http://zh.lucida.me/blog/java-8-lambdas-insideout-l ...

  9. paip。java 高级特性 类默认方法,匿名方法+多方法连续调用, 常量类型

    paip.java 高级特性 类默认方法,匿名方法+多方法连续调用, 常量类型 作者Attilax 艾龙,  EMAIL:1466519819@qq.com 来源:attilax的专栏 地址:http ...

随机推荐

  1. [bzoj3527][Zjoi2014]力_FFT

    力 bzoj-3527 Zjoi-2014 题目大意:给定长度为$n$的$q$序列,定义$F_i=\sum\limits_{i<j}\frac{q_iq_j}{(i-j)^2}-\sum\lim ...

  2. hibernate详解一

    hibernate介绍 hibernate是一个开源的轻量级的框架, hibernate框架应用在javaee三层结构中的dao层框架,在dao层对数据库进行crud操作,使用hibernate框架实 ...

  3. fetch各种报跨域错误,数据无法获取的解决方案

    1.介绍 fetch 提供了一个获取资源的接口 (包括跨域). fetch 的核心主要包括:Request , Response , Header , Body 利用了请求的异步特性 --- 它是基于 ...

  4. 【Nginx】负载均衡-加权轮询策略剖析

    转自:江南烟雨 本文介绍的是客户端请求在多个后端服务器之间的均衡,注意与客户端请求在多个nginx进程之间的均衡相区别. 如果Nginx是以反向代理的形式配置运行,那么对请求的实际处理需要转发到后端服 ...

  5. SaltStack学习系列之State安装Nginx+PHP环境

    目录结构 |-- pillar | |-- nginx | | `-- nginx.sls #nginx变量(key:value) | `-- top.sls `-- salt|-- init #初始 ...

  6. ISkyShop B2B2C 商城系统V1.0正式版隆重公布

    ISkyShop核心开发团队结合7年电商开发经验,历经1年多时间的设计研发,于2014年6月12日隆重推出ISkyShop B2B2C 商城系统V1.0,B2B2C商城系统是ISkyShop独立自主研 ...

  7. Jenkins安装与使用

    一.Jenkins简介 Jenkins是基于Java开发的一种持续集成工具,用于监控持续重复的工作,功能包括: 1.持续的软件版本发布/测试项目. 2.监控外部调用执行的工作 二.下载与安装 下载地址 ...

  8. Linux 命令修改系统时间

    修改linux的系统时间使用date指令,date命令的功能是显示和设置系统日期和时间. 输入date 查看目前系统时间. 修改时间需要 date -功能字符 修改内容 命令中各选项的含义分别为:-d ...

  9. JavaScript 获得代码行号和脚本文件名

    如果你使用的是 V8 引擎,Chrome 和 Node.js 所用的,那么你可以利用 JavaScriptStackTraceApi 来获得行号信息,有两个 API: Error.captureSta ...

  10. Android框架之高速开发框架xUtil

    做Android开发我们通常是从原生态的開始,就是调用默认那些Android代码来开发我们的应用,可是到了一定程度,我们就想着怎么来高速开发我们的应用.这个时候我们就要着手来研究框架了. 以下介绍一个 ...