您的位置:首页 > 编程语言 > Java开发

JAVA8接口中的default、static方法使用注意事项

2017-04-16 19:58 459 查看

JAVA8中接口interface引入方法实现的猜想

在JAVA8之前java中的接口类型旨在定义类型的接口规约,并不能拥有自己的方法实现,具体实现完全交由实现类自己完成。

以普通人和中上阶层为例,为各自的子女读大学的目标:

普通人大部分甚至可能全部都是告诫子女读书毕业找份好工作;

后者更多要求考TOEFL、GRE、出国留学回来做高管、出资给孩子创业;

接口规约的方法行为是宽泛的,无法具体实现;

而现实中的实现则是 符合该规约的实体自己的实现。

但是在JAVA8中可以在接口中定义默认的行为、以及静态方法的实现。

你可以想象,人类文明发展到后面,大家都不需要工作了,完全交由机器人处理。这时候,人们的行为是一样的: 出生 – 接受各种知识注入脑子里–不工作–等待生命结束(或者永生)。 规约可以有一个稍微具体的实现。大家都可以使用该具体的行为。这就是JAVA8中的default、static方法。

default方法

java.lang.Iterable接口中有一个默认的方法实现:

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


default方法使用default关键字修饰,它是对象方法,需要使用对象来进行访问。

以下的示例中,使用了JAVA8中的新注解 @FunctionalInterface 表明该接口是一个函数式接口,只能拥有 一个抽象方法。

package com.byron4j.hightLevel.java8.lambda;

/**
*
* <pre>
*      接口类型 拥有自己的default、static方法实现
*      @FunctionalInterface 表明该接口是一个函数式接口,只能拥有 一个抽象方法
* </pre>
* @author Byron.Y.Y
*/

@FunctionalInterface
public interface DefaultStaticMethodDemo {

/*非default、static方法不能有实现
* --否则编译错误--Abstract methods do not specify a body
void sayHello4CompilerError(){};
*/

void sayHello();

/*default、static方法必须有具体的实现
* --否则编译错误--This method requires a body instead of a semicolon
default void studyTarget();
*/

default void studyTarget(){
System.out.println("出生");
System.out.println("\t--> 注入知识");
System.out.println("\t\t--> 生命消亡");
}

//可以拥有多个default方法
default void studyTarget2(){
System.out.println("DefaultStaticMethodDemo#【default】studyTarget2 invok.");
}

//可以拥有多个static方法
static void info(){
System.out.println("DefaultStaticMethodDemo#【static】 info invok.");
}

public static void main(String[] args) {
info();

new DefaultStaticMethodDemo() {
//仅仅需要实现抽象方法
//default、static方法不需要强制自己新实现
@Override
public void sayHello() {
// TODO Auto-generated method stub

}
};
}
}


接口中方法使用的注意事项

非default、static方法不能有实现,否则编译错误:Abstract methods do not specify a body

default、static方法必须有具体的实现,否则编译错误:This method requires a body instead of a semicolon

可以拥有多个default方法

可以拥有多个static方法

使用接口中类型时,仅仅需要实现抽象方法,default、static方法不需要强制自己新实现

实现多个接口引发的问题

多个接口存在签名一样的default方法导致编译错误

JAVA8中实现多个接口可能引发的问题:多个接口存在签名一样的default方法导致编译错误。

又有一个接口DefaultStaticMethodDemo2 和DefaultStaticMethodDemo都拥有一样的签名的default studyTarget方法:

package com.byron4j.hightLevel.java8.lambda;

/**
*
* <pre>
*      接口类型 拥有自己的default、static方法实现
*      @FunctionalInterface 表明该接口是一个函数式接口,只能拥有 一个抽象方法
* </pre>
* @author Byron.Y.Y
*/

public interface DefaultStaticMethodDemo2 {

default void studyTarget(){
System.out.println("出生");
System.out.println("\t--> 注入知识");
System.out.println("\t\t--> 生命消亡");
}

}


因为可以实现多个接口,所以类SubClassDemo实现了DefaultStaticMethodDemo、DefaultStaticMethodDemo2,但是编译不通过:

package com.byron4j.hightLevel.java8.lambda;

//编译错误:Duplicate default methods named studyTarget
//  with the parameters () and () are inherited from the
//types DefaultStaticMethodDemo2 and DefaultStaticMethodDemo
public class SubClassDemo implements
DefaultStaticMethodDemo, DefaultStaticMethodDemo2{

}


这就相当于你在类中定义了两个同样的签名方法,从而引致编译错误。

继承抽象类同时实现接口引发的问题

如果一个类实现了某个拥有default方法的接口的话,在该类中则不需要自己再次实现该default方法了

但是如果该类实现接口时,还继承了某个抽象类,该抽象类拥有一个和default签名一样的抽象方法,则在该类中必须重写抽象方法(也是接口中的该default方法)

抽象类AbstractClassDemo 拥有和接口DefaultStaticMethodDemo同安用的签名方法studyTarget

package com.byron4j.hightLevel.java8.lambda;

public abstract class AbstractClassDemo {
abstract void studyTarget();
}


类SubClassDemo2 必须重写studyTarget方法

package com.byron4j.hightLevel.java8.lambda;

/**
*
* <pre>
*      继承抽象类
*      实现接口
*      抽象类、接口存在同样的签名方法
*      抽象类未有实现体;接口中default实现了方法。
* </pre>
* @author Byron.Y.Y
*/
public class SubClassDemo2 extends AbstractClassDemo implements
DefaultStaticMethodDemo {

@Override
public void sayHello() {
// TODO Auto-generated method stub

}

@Override
public void studyTarget() {
// TODO Auto-generated method stub

}

}


其他注意事项

抽象类、接口存在同样的签名方法,抽象类有实现体但是不是public修饰的;—-> 编译错误:抽象接口中的实现不能隐藏接口中的方法;—->解决办法:将抽象类中的方法访问控制符使用public修饰。

package com.byron4j.hightLevel.java8.lambda;

public abstract class AbstractClassDemo2 {
void sayHello(){
System.out.println("抽象类的sayHello!");
}
}


package com.byron4j.hightLevel.java8.lambda;

/**
*
* <pre>
*      继承抽象类
*      实现接口
*      抽象类、接口存在同样的签名方法
*      抽象类有实现体但是不是public修饰的;
*-------------编译错误:抽象接口中的实现不能隐藏接口中的方法
*-------------解决办法:将抽象类中的方法访问控制符使用public修饰
* </pre>
* @author Byron.Y.Y
*/
public class SubClassDemo3 extends AbstractClassDemo2 implements
DefaultStaticMethodDemo {

}


抽象类AbstractClassDemo2 拥有和接口DefaultStaticMethodDemo相同签名的方法sayHello,但是AbstractClassDemo2 的实现不是public的引发编译错误。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: