Java 8 中的设计模式策略

2022-05-27 01:09:32 浏览数 (1)

概述

在本篇文章中我们对可以在 Java 8 中的设计模式策略(strategy design pattern)进行一些简单的说明。

如果你对 Java 的设计模式不是非常清楚的话,可以先自行脑补下。

我们简单的总结就是将以前 Java 使用的接口和实现的设计模式,在 Java 8 中可以使用 lambda 函数来进行简化。

在下面内容中,我们首先提供了一个简单的设计模式样例,以及在传统的环境下我们是怎么实现这个设计模式的。

随后,我们将会使用 Java 8 中的 lambda 函数来进行实现,然后介绍一些有什么不同的地方。

模式策略

所谓的模式策略(strategy pattern)的定义就是能够让我们的程序在运行时(runtime)改变算法的表现。

在通常的情况下,我们会首先设计一个接口,然后在这个接口中定义我们需要使用的方法,然后使用不同的类来实现我们的接口定义的方法。

这种设计模式为我们在 Java 面向对象设计时候经常用到的。

让我们来考察下面的一个使用案例,针对不同的节日,我们针对某一个销售使用不同的定价策略,比如说圣诞节(Christmas),复活节(Easter)或者新年(New Year),我们使用的价格策略是不一样的。

首先我们需要在接口中定义一个 Discounter 方法,然后针对不同的节日来实现 Discounter 这个方法。

代码语言:javascript复制
public interface Discounter {
    BigDecimal applyDiscount(BigDecimal amount);
}

然后我们的目标是在复活节的时候打 5 折(50%),另外一个目标是在圣诞节的时候打 9 折(10%)。

随后我们就可以在下面的 2 个类中实现我们在接口中定义的方法。

代码语言:javascript复制
public static class EasterDiscounter implements Discounter {
    @Override
    public BigDecimal applyDiscount(final BigDecimal amount) {
        return amount.multiply(BigDecimal.valueOf(0.5));
    }
}

public static class ChristmasDiscounter implements Discounter {
   @Override
   public BigDecimal applyDiscount(final BigDecimal amount) {
       return amount.multiply(BigDecimal.valueOf(0.9));
   }
}

然后,我们在则是中使用这个策略:

代码语言:javascript复制
Discounter easterDiscounter = new EasterDiscounter();

BigDecimal discountedValue = easterDiscounter
  .applyDiscount(BigDecimal.valueOf(100));

assertThat(discountedValue)
  .isEqualByComparingTo(BigDecimal.valueOf(50));

上面这个设计模式是我们在通常情况下使用的,但是比较头痛的是针对每一个方法,你需要在实现中都实现你需要的方法。

另外一个解决方案就是使用内部类型,但是这个内部类型并没有有太多的提高,你还是有不少的工作需要做。

例如下面使用内部类型的实现:

代码语言:javascript复制
Discounter easterDiscounter = new Discounter() {
    @Override
    public BigDecimal applyDiscount(final BigDecimal amount) {
        return amount.multiply(BigDecimal.valueOf(0.5));
    }
};

使用 Java 8

如果你开始使用 Java 8 的话,我们知道 lambda 函数表达式可以做内部类型来使用,这样能够明显的降低多余的代码。

同时会让我们的代码看起来更加整洁和可读。

不管怎么样,使用 lambda 表达是提供了另外一种模式的实现,针对最开始的实现来说,Java 8 的实现提供了更多的一种选择。

降低代码的冗余

现在我们针对 EasterDiscounter 的实现,我们现在只是用 lambda 表达式来实现:

代码语言:javascript复制
Discounter easterDiscounter = amount -> amount.multiply(BigDecimal.valueOf(0.5));

通过上面的代码,我们可以看到使用 lambda 表达式的实现看起来更加整洁,代码更加可读和便于维护,针对开始使用多行才能实现的内容,现在只需要使用一行就可以完成了。

更主要的是: ** 一个 lambda 表达式可以被用来替换匿名的内部类型**。

如果我们需要对多个折扣力度进行实现的话,使用 lambda 表达式就看起来更加漂亮了:

代码语言:javascript复制
List<Discounter> discounters = newArrayList(
  amount -> amount.multiply(BigDecimal.valueOf(0.9)),
  amount -> amount.multiply(BigDecimal.valueOf(0.8)),
  amount -> amount.multiply(BigDecimal.valueOf(0.5))
);

如果我们需要对很多折扣力度进行定义的话,我们可以在 Java 8 中使用静态方法,然后这个定义将会在一个类中完成。

如果你愿意的话,Java 8 甚至可以让你在接口中定义静态方法。

对比在实体类和匿名内部类型之间进行选择,让我们在一个单独类中创建多个静态 lambda 表达式:

代码语言:javascript复制
public interface Discounter {
    BigDecimal applyDiscount(BigDecimal amount);

    static Discounter christmasDiscounter() {
        return amount -> amount.multiply(BigDecimal.valueOf(0.9));
    }

    static Discounter newYearDiscounter() {
        return amount -> amount.multiply(BigDecimal.valueOf(0.8));
    }

    static Discounter easterDiscounter() {
        return amount -> amount.multiply(BigDecimal.valueOf(0.5));
    }
}

通过上面的代码,我们可以看到使用了较少的代码,我们实现了很多的功能。

改进方法的创建

让我们来对 Discounter 接口再次进行修改,这次我们让 Discounter 接口继承 UnaryOperator 接口,然后添加一 combine() 方法:

代码语言:javascript复制
public interface Discounter extends UnaryOperator<BigDecimal> {
    default Discounter combine(Discounter after) {
        return value -> after.apply(this.apply(value));
    }
}

最开始的设计就是通过对 Discounter 接口的调整,能够让 Discounter 接口能够对折扣进行处理。

随着 UnaryOperator 接口被继承,我们可以使用 UnaryOperator 接口提供的 apply() 方法,我们只需要对 applyDiscount 进行替换就可以了。

combine() 方法为在 Discounter 接口中应用的一个抽象,使用一个内建的 apply() 函数来实现。

现在,让我们来试试针对某一个价格实现多折扣的情况,我们将会使用 reduce() 和我们的 combine() 函数来实现:

代码语言:javascript复制
Discounter combinedDiscounter = discounters
  .stream()
  .reduce(v -> v, Discounter::combine);

combinedDiscounter.apply(...);

特别关注下 reduce 的第一参数,如果我们没有任何折扣被使用,我们需要返回一个没有修改的值。

当然你也可以在这里定义一个函数,通过这个定义的函数来实现一个默认的折扣。

针对默认的遍历选项来说,通过这种实现为我们提供了更多的函数功能。

结论

在本代码中,我们对 Java 8 中的设计模式策略(strategy design pattern)进行一些简单的说明,因为 lambda 表达式的使用,让我们能够使用更少的代码实现更多的功能。

如果你觉得这篇文章有点难度的话,你需要先对 Java 的一些面向对象设计有所了解,对接口,抽象,继承,内部类型等都需要有些熟悉才能更好的理解。

https://www.ossez.com/t/java-8/13978

0 人点赞