Java8特性第四讲:Java 8的接口默认方法实现

Java8特性第四讲:Java8的接口默认方法实现

本文是Java8特性第四讲,主要讲解Java 8的接口默认方法实现。理解Java 8 默认方法需理解几个问题: 1、为什么会出现默认方法? 2、接口中出现默认方法,且类可以实现多接口的,那和抽象类有啥区别? 3、多重实现的默认方法冲突怎么办?

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

1.1、案例

一个接口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()方法的默认实现。

1.2、什么是默认方法

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

1.3、为什么出现默认方法

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

背景:Java8中给很多API提供了新的方法,比如常用的java.util.List接口增加了replaceAll、sort、spliterator这三个方法;根据现行的Java语法规范,List的所有子类都需要对这三个方法进行实现,这显然是不能接受的,尤其是众多第三方类库对JDK有依赖,要求他们一同修改也是做不到的

  • 为了解决这个问题,也保证JDK的兼容性,Java8引入了一种新的机制:接口可以声明带有实现的方法

语法规则

  • 1、使用default关键字
  • 2、一个接口中可以含有多个带有默认实现的方法

java.util.List中新增的三个方法源码

public interface List<E> extends Collection<E> {
    default void replaceAll(UnaryOperator<E> operator) {
        Objects.requireNonNull(operator);
        final ListIterator<E> li = this.listIterator();
        while (li.hasNext()) {
            li.set(operator.apply(li.next()));
        }
    }
    
    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 Spliterator<E> spliterator() {
        return Spliterators.spliterator(this, Spliterator.ORDERED);
    }
}

2、java 8抽象类与接口对比

这一个功能特性出来后,有人反馈,java 8的接口都有实现方法了,跟抽象类岂不是差不多?其实还是有的,请看下表对比。。

相同点不同点
都是抽象类型抽象类不可以多重继承,接口可以(无论是多重类型继承还是多重行为继承)
都可以有实现方法(以前接口不行)抽象类和接口所反映出的设计理念不同。其实抽象类表示的是”is-a”关系,接口表示的是”like-a”关系
都可以不需要实现类或者继承者去实现所有方法,(以前不行,现在接口中默认方法不需要实现者实现)接口中定义的变量默认是public static final 型,且必须给其初值,所以实现类中不能改变其值;抽象类中的变量默认是 friendly 型,其值可以在子类中重新定义,也可以重新赋值。

3、多重继承的冲突

由于同一个方法可以从不同接口引入,自然而然的会有冲突的现象,同一个类可能会从不同的接口中继承了具有相同的签名的方法,这又该如何解决呢?

默认方法判断冲突的规则如下:

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

2.否则,则会优先选取路径最短的。

3.1、举例子

场景1:

public interface A {
    default String test() {
        return "A";
    }
}
public class B {
    public String test() {
        return "B";
    }
}
public class C extends B implements A {
    public static void main(String[] args) {
        System.out.println(new C().test());
    }
}
  • 这个例子中C分别中A和B中都继承了test()方法,但是B是类,而A是接口,所以B中的test()方法优先级更高,因此此次输出结果应该是:B

场景二:

public interface D extends A {
    default String test() {
        return "D";
    }
}
public class E implements A, D {
    public static void main(String[] args) {
        System.out.println(new E().test());
    }
}
  • 这个例子中E分别中A和D这两个接口中继承了test()方法,但是D是A的子接口,所以D接口中的test()方法优先级更高,因此此处输出应该是:D

场景三:

public class F implements A {
    
}
public class G extends F implements A, D {
    public static void main(String[] args) {
        System.out.println(new G().test());
    }
}
  • 这种场景下虽然F实现了A接口,但是由于它并没有重写test()方法,所以优先级依然是D接口高,因此此处输出依然是:D

场景四:

public interface H {
    default String test() {
        return "H";
    }
}
public class K implements A, H {
    public static void main(String[] args) {
        System.out.println(new K().test());
    }
    
    @Override
    public String test() {
        return H.super.test();
    }
}
  • 这个场景中,由于A和H没有继承关系,所以在JVM无法识别出到底要调用哪一个接口中的方法,所以需要在类K中显式的覆盖test()方法,指明具体调用哪个接口中的方法,当然也可以自己重写方法实现;

4、默认方法在项目中的使用

场景1:在规则引擎实现类中,我们使用默认方法标识执行阶段,这样子类不用重写该方法,可以节省代码量。

/**
 * 业务检查
 **/
public interface BizChecker extends Checker {
    /**
     * 适用什么阶段
     *
     * @return
     */
    @Override
    @NotNull
    default PhaseEnum phase() {
        return PhaseEnum.BIZ_CHECK;
    }
}

场景2:在错误码接口中,将获取错误字符串方法设置为默认方法,可以节省代码量。

/**
 * 错误码接口
 **/
public interface ErrorCode {

    /**
     * 错误码 必须在{@link ErrorCodeEnum} 中定义
     * @return
     * @see {@link ErrorCodeEnum#getErrCode()}
     */
    String getErrCode();

    /**
     * 错误描述
     * @return
     */
    String getErrDesc();

    /**
     * 是否用在response里对外提示
     * @return
     */
    boolean isFirstTip();

    /**
     * 获取错误字符串:code+desc
     * @return
     */
    default String getErrString() {
        return "code:" + getErrCode() + ",desc:" + getErrDesc() + ",useInResp:" + isFirstTip();
    }
}

场景3:限流器默认方法

/**
 * @Description 限流器
 */
public interface RateLimiter {
    default void acquire() {
        acquire(1);
    }

    /**
     * 限流
     * @param permits 请求资源数
     */
    void acquire(int permits);
}

场景4:处理商品限价,默认方法放在接口上

public interface GoodsFirmHighestPricePretreated {

  	/**
     * 处理最高限价针对商品简化
     */
    default Response<Boolean> handleFirmHighestPrice(FullItemDTO fullItemDTO, GoodsPriceContext context) {
        if (CollectionUtils.isEmpty(fullItemDTO.getBaseItem().getSkus())) {
            return Response.ok(true);
        }
    }
}

5、总结

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

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: Java 8是一个非常重要的版本,它提供了许多新的特性和功能。这里是一些Java 8的新特性的案例: 1. Lambda表达式:Lambda表达式是一种新的匿名函数,允许您直接在代码中传递函数。例如,可以使用lambda表达式来创建并运行线程: ``` Thread t = new Thread(() -> System.out.println("Hello from a thread!")); t.start(); ``` 2. Stream API:Stream API是Java 8中的一个新功能,允许您使用函数式编程风格处理数据集合。例如,可以使用Stream API计算列表中所有数字的平均值: ``` List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5); double average = numbers.stream().mapToInt(i -> i).average().getAsDouble(); System.out.println("Average: " + average); ``` 3. 接口默认方法:在Java 8中,接口可以有默认方法,即在接口中定义的带有实现方法。这允许在不修改已实现接口的类的情况下向接口添加新方法。例如,可以定义一个接口来表示形状,并在接口中定义一个默认方法来计算形状的面积: ``` public interface Shape { double getArea(); default double getPerimeter() { return 0; } } public class Circle implements Shape { private double radius; public Circle(double radius) { this.radius = radius; } @Override public double getArea() { return Math.PI * radius * radius; } } ``` 这些只是Java 8的一些新特性的例子。 ### 回答2: Java 8是Java编程语言的一个重要版本,引入了许多新特性和改进,以下是一些Java 8的新特性案例。 1. Lambda表达式:Java 8引入了Lambda表达式,可以更简洁地编写函数式接口实现。例如,使用Lambda表达式实现Comparator接口进行字符串排序可以写作:List<String> list = Arrays.asList("c", "a", "b"); list.sort((s1, s2) -> s1.compareTo(s2)); 2. Stream API:Stream API提供了一种新的抽象数据类型,用于操作集合和数组。通过使用Stream API,可以非常方便地进行集合的过滤、转换和操作。例如,计算集合中所有偶数的平方和可以写作:List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5); int sum = numbers.stream().filter(n -> n % 2 == 0).mapToInt(n -> n * n).sum(); 3. Optional类:Optional类是一种用于处理可能不存在的值的容器类。它可以避免空指针异常,提供更好的代码可读性。例如,使用Optional类可以更安全地获取一个可能为空的字符串长度:Optional<String> optional = Optional.ofNullable(str); int length = optional.map(String::length).orElse(0); 4. 新的日期/时间API(Date and Time API):Java 8引入了全新的日期/时间API,提供了更强大和易用的日期和时间处理功能。新的日期/时间API提供了LocalDate、LocalTime、LocalDateTime等类,用于处理日期、时间和日期时间。例如,获取当前日期时间可以写作:LocalDateTime now = LocalDateTime.now(); 5. 并发增强:Java 8引入了新的并发工具,如CompletableFuture、Stream和ParallelStream等,用于方便地实现并发编程。例如,使用ParallelStream执行并行计算可以提高性能:IntStream.range(1, 10).parallel().forEach(System.out::println); 这些是Java 8的一些新特性案例,这些新特性使得Java编程更加简洁、高效和易用,提供了更多的编程选择和便利。 ### 回答3: Java 8是Java编程语言的一个重大更新版本,引入了许多令人兴奋的新特性。下面我将举几个例子来说明Java 8的一些新特性。 首先,Java 8引入了Lambda表达式。Lambda表达式使得我们能够以一种更简洁和声明式的方式编写代码。例如,在过去,我们经常需要使用匿名内部类来实现某个接口方法,现在我们可以使用Lambda表达式来代替这种冗长的写法。比如,我们可以通过以下方式来实现一个Runnable接口的run方法: ``` Runnable runnable = () -> { System.out.println("Hello, World!"); }; ``` 第二个例子是函数式接口。函数式接口是一个只包含一个抽象方法接口,它可以被Lambda表达式所实现Java 8提供了许多内置的函数式接口,如Consumer、Function和Predicate。这些函数式接口使得我们能够以更直观的方式处理集合和流操作。例如,我们可以使用Predicate接口来筛选一个集合中的元素: ``` List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10); List<Integer> evenNumbers = numbers.stream() .filter(n -> n % 2 == 0) .collect(Collectors.toList()); ``` 第三个例子是Stream API。Stream API提供了一种新的方式来处理集合数据,它引入了一些类似于SQL查询的操作,如map、filter、reduce和collect等。Stream API使得我们能够以一种更优雅和简单的方式进行复杂的集合操作。例如,我们可以使用Stream API来计算一个集合中所有元素的和: ``` List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10); int sum = numbers.stream() .reduce(0, Integer::sum); ``` 总之,Java 8引入了许多令人兴奋的新特性,如Lambda表达式、函数式接口和Stream API。这些新特性使得我们能够以一种更简洁、高效和优雅的方式编写Java代码,提高代码的可读性和可维护性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

程序员 jet_qi

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值