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




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);
        }
    }
  
  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

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

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

扫描二维码关注公众号,回复: 1782026 查看本文章
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

            }
        };
    }
}

  
  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62

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

  • 非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--> 生命消亡");
    }

}

  
  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23

因为可以实现多个接口,所以类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{

}

  
  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

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

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

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

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

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

package com.byron4j.hightLevel.java8.lambda;

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

  
  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

类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

    }

}

  
  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30

其他注意事项

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

package com.byron4j.hightLevel.java8.lambda;

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

  
  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
package com.byron4j.hightLevel.java8.lambda;



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



}

  
  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23

抽象类AbstractClassDemo2 拥有和接口DefaultStaticMethodDemo相同签名的方法sayHello,但是AbstractClassDemo2 的实现不是public的引发编译错误。






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

猜你喜欢

转载自blog.csdn.net/weixin_39778570/article/details/80828649
今日推荐