文档章节

Java 8新特性探究(二)深入解析默认方法

OSC闲人
 OSC闲人
发布于 2013/11/13 15:12
字数 1065
阅读 11877
收藏 137

上篇讲了 lambda表达式的语法,但只是 JEP126 特性的一部分,另一部分就是默认方法(也称为虚拟扩展方法或防护方法)

什么是默认方法,为什么要有默认方法

简单说,就是接口可以有实现方法,而且不需要实现类去实现其方法。只需在方法名前面加个default关键字即可。

为什么要有这个特性?首先,之前的接口是个双刃剑,好处是面向抽象而不是面向具体编程,缺陷是,当需要修改接口时候,需要修改全部实现该接口的类,目前的java 8之前的集合框架没有foreach方法,通常能想到的解决办法是在JDK里给相关的接口添加新的方法及实现。然而,对于已经发布的版本,是没法在给接口添加新方法的同时不影响已有的实现。所以引进的默认方法。他们的目的是为了解决接口的修改与现有的实现不兼容的问题。

简单的例子

一个接口A,Clazz类实现了接口A。

public interface A {
    default void foo(){
       System.out.println("Calling A.foo()");
    }
}
  
public class Clazz implements A {
    public static void main(String[] args){
       Clazz clazz = new Clazz();
       clazz.foo();//调用A.foo()
    }
}

代码是可以编译的,即使Clazz类并没有实现foo()方法。在接口A中提供了foo()方法的默认实现。

java 8抽象类与接口对比

这一个功能特性出来后,很多同学都反应了,java 8的接口都有实现方法了,跟抽象类还有什么区别?其实还是有的,请看下表对比。。

相同点 不同点

1.都是抽象类型;

2.都可以有实现方法(以前接口不行);

3.都可以不需要实现类或者继承者去实现所有方法,(以前不行,现在接口中默认方法不需要实现者实现)

1.抽象类不可以多重继承,接口可以(无论是多重类型继承还是多重行为继承);

2.抽象类和接口所反映出的设计理念不同。其实抽象类表示的是"is-a"关系,接口表示的是"like-a"关系;

3.接口中定义的变量默认是public static final 型,且必须给其初值,所以实现类中不能改变其值;抽象类中的变量默认是 friendly 型,其值可以在子类中重新定义,也可以重新赋值。 

多重继承的冲突说明

由于同一个方法可以从不同接口引入,自然而然的会有冲突的现象,默认方法判断冲突的规则如下:

1.一个声明在类里面的方法优先于任何默认方法(classes always win)

2.否则,则会优先选取最具体的实现,比如下面的例子 B重写了A的hello方法。

输出结果是:Hello World from B

如果想调用A的默认函数,则用到新语法X.super.m(...),下面修改C类,实现A接口,重写一个hello方法,如下所示:

public class C implements A{
   
    @Override
    public void hello(){
        A.super.hello();
    }
    
    public static void main(String[] args){
        new C().hello();
    }
}

输出结果是:Hello World from A

总结

默认方法给予我们修改接口而不破坏原来的实现类的结构提供了便利,目前java 8的集合框架已经大量使用了默认方法来改进了,当我们最终开始使用Java 8的lambdas表达式时,提供给我们一个平滑的过渡体验。也许将来我们会在API设计中看到更多的默认方法的应用。

跟上篇博文结合起来,就是JEP126的全部了,后面还有54个特性等着我们去探究,为了让大家比较深刻了解lambda,学以致用,下一篇还是lambda的内容,预告一下下篇的标题:《Java 8特性探究(三)解开lambda表达式最强作用的神秘面纱》,第二个特性 将从第四篇开始,谢谢大家支持,敬请期待。。。


转载时候请注明出处。 http://my.oschina.net/benhaile


© 著作权归作者所有

共有 人打赏支持
OSC闲人

OSC闲人

粉丝 3547
博文 22
码字总数 26172
作品 1
深圳
技术主管
加载中

评论(49)

管龙
管龙
一篇文章是干嘛的?为什么要加这个特性?怎么加?怎么做?说的真是好啊再次留言!
管龙
管龙
写的真是好啊 读了三行外加第一篇 佩服的五体投地!能加个QQ吗?或者你有什么群拉一个啊 一个小小程序员!我的Q 897771860
java9
java9

引用来自“yfn020”的评论

错误: 默认超级调用中的类型限定符A错误
A.super.hello();
^
冗余接口 A 已由 B 扩展
1 个错误

java -version
java version "1.8.0"
Java(TM) SE Runtime Environment (build 1.8.0-b132)
Java HotSpot(TM) 64-Bit Server VM (build 25.0-b7
这种我也编绎不通过
悠槿璃
悠槿璃
……
beanGou
beanGou
文章末尾,要调用A的default方法,C不用可以重写这个方法了, 直接调用A的方法好啦
末秋弦影
不好意思!43楼有误。。看错了,是接口名称.super.默认方法,可以用了,但是必须在实现了该接口的类中才被允许使用13
末秋弦影
类名.super.默认方法 只有在该类只实现了单一接口时才能用,如果该类实现了多个接口就会报错。
这个应该解决不了吧?
OSC闲人
OSC闲人

引用来自“cwalet”的评论

public class TestInterface implements A, B {

public void foo() {
System.out.println("test");
A.super.foo();
}

public static void main(String[] args) {
new TestInterface().foo();
}
}

这段代码会编译出错:冗余接口A已由B扩展。

也就是说只要实现了子接口,那么父接口的同名默认方法是不可能被调用到的?
A,B都有相同的默认方法了
cwalet
cwalet
public class TestInterface implements A, B {

public void foo() {
System.out.println("test");
A.super.foo();
}

public static void main(String[] args) {
new TestInterface().foo();
}
}

这段代码会编译出错:冗余接口A已由B扩展。

也就是说只要实现了子接口,那么父接口的同名默认方法是不可能被调用到的?
Sky__
Sky__

引用来自“Sky__”的评论

3.接口中定义的变量默认是public static final 型,且必须给其初值,所以实现类中不能重新定义
实现类中应该可以重新定义吧,如下:
package foo2;

interface Human {
  String DEF_NAME="地球人";
  
}

class Student implements Human{
  String DEF_NAME="学生";
  
}

public class MainRun{
  
  public static void main(String args[]){
    //Human h=new Student();
    System.out.println(Human.DEF_NAME);
    
    Student h2=new Student();
    System.out.println(h2.DEF_NAME);
  }
}

引用来自“成熟的毛毛虫”的评论

不对。。Student并不是修改了Human的DEF_NAME的值,可以加个构造函数就知道了 public Student(){ Human.DEF_NAME="学生" } IDE会提示不能修改final值,不过“重新定义”这个有歧义,应该是“重新修改”
嗯,谢谢提示! 上班在学习别人写的博客,没改bug,表示会不会被强撸啊? 呵呵
Java 8新特性探究(三)解开lambda最强作用的神秘面纱

我们期待了很久lambda为java带来闭包的概念,但是如果我们不在集合中使用它的话,就损失了很大价值。现有接口迁移成为lambda风格的问题已经通过default methods解决了,在这篇文章将深入解析...

OSC闲人
2013/11/18
0
42
What's New in JDK 8 java8新特性汇总

Oracle甲骨文公司终于在2014年3月发布了Java 8正式版,它是java的一个里程牌版本,带来了诸多新特性。 针对这些新特性汇总如下: 一、针对java编程语言(Java Programming Language) 1.lambd...

cloud-coder
2014/04/03
0
4
Java 8新特性探究 前言

自2013年6月13日,oracle就已经发布的Java 8特性完备版本(M7),但最终GA版本将在2014年3月18日(已第二次跳票,原计划今年9月发布的,据官网宣称是为了解决安全问题......),相信大家多多...

OSC闲人
2013/11/05
0
17
Java 8新特性探究(五)重复注解(repeating annotations)

知识回顾 前面介绍了: lambda表达式和默认方法 (JEP 126) 批量数据操作(JEP 107) 类型注解(JEP 104) 注:JEP=JDK Enhancement-Proposal (JDK 增强建议 ),每个JEP即一个新特性。 在j...

OSC闲人
2013/12/03
0
1
一起学Java7新功能扩展——深入历险分享(一)

特此声明:因网友疑问,这里声明一个重要的安全,就是大家所知的java惊现0day漏洞!8月30日,Oralce紧急发布了新版本的JDK和JRE,原因是发现了一个严重的0day漏洞CVE-2012-4681,远程攻击者可...

Beyond-Bit
2012/09/03
0
26

没有更多内容

加载失败,请刷新页面

加载更多

下一页

c语言之内存分配笔记

先看一个数组: short array[5] = {1,2} // 这儿定义的一个int类型的数组,数组第1和第2个元素值是1和2.其余后面默认会给值为0; 或者 short array[] = {1,2};//这儿数组第1和第2个元素,数组...

DannyCoder
今天
2
0
Shell | linux安装包不用选择Y/N的方法

apt-get install -y packageOR echo "y" | sudo apt-get install package

云迹
今天
2
0
Hadoop的大数据生态圈

基于Hadoop的大数据的产品圈 大数据产品的一句话概括 Apache Hadoop: 是Apache开源组织的一个分布式计算开源框架,提供了一个分布式文件系统子项目(HDFS)和支持MapReduce分布式计算的软件架...

zimingforever
今天
5
0
八大包装类型的equals方法

先看其中一个源码 结论:八大包装类型的equals方法都是先判断类型是否相同,不相同则是false,相同则判断值是否相等 注意:包装类型不能直接用==来等值比较,否则编译报错,但是数值的基本类型...

xuklc
今天
2
0
NoSQL , Memcached介绍

什么是NoSQL 非关系型数据库就是NoSQL,关系型数据库代表MySQL 对于关系型数据库来说,是需要把数据存储到库、表、行、字段里,查询的时候根据条件一行一行地去匹配,当量非常大的时候就很耗...

TaoXu
昨天
5
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部