Java设计模式-策略模式

将帅谋略,知兵者,动而不迷,举而不穷。

策略模式的定义

所谓策略模式,即

定义一系列的算法,把它们一个个封装起来, 并且使它们可相互替换。

通俗的一点来讲,当同一个操作,可能有不同的表现或者动作的时候,我们可以将这个操作抽象出来,然后将具体的表现或者动作分别的封装起来,并且都实现抽象的操作,这样我们就可以在执行这个操作的时候,根据不同的表现类型,来执行不同表现或者动作。

策略模式结构可分为,策略接口,具体策略实现和策略容器。

  • 策略接口用于抽象定义具体的操作。
  • 策略实现用于定义不同策略下的具体实现。
  • 策略容器用于存放策略和执行策略。

关系图如下:

策略模式

策略模式的场景

我们在电商平台购物,到支付支付的时候,通常会有很多的支付方式可以选择,比如说有:支付宝、微信、银联云闪付等等。

这里我们就可以思考一下,电商平台在执行支付这个操作的时候,是如何根据我们选择的支付方式来判断和执行之后的操作的呢?

最容易想到的,可能就是在代码中根据用户选择的支付方式进行一系列的 if else 操作了把。

if (Objects.equals(paymentWay, "支付宝")) {
    支付宝支付处理逻辑...
} else if (Objects.equals(paymentWay, "微信")) {
    微信支付处理逻辑...
} else if (Objects.equals(paymentWay, "银联云闪付")) {
    银联云闪付支付处理逻辑...
} else if (Objects.equals(paymentWay, "xxx")) {
    xxx支付处理逻辑...
}

看看上面的代码是不是觉得头晕胸闷。每个 if else 中都有大量的逻辑处理,新增一个支付方式,就需要新增一个 if else ,这样写了之后,不仅代码的可读性低,而且后期的维护成本也是大大的增高了。而且每次的修改出错的几率也是比较高。

所以这里就适合使用策略模式来进行一波重构。

策略模式的例子

这个例子,就以上面说的支付来举例。

我们可以把支付这个操作抽象为一个策略接口。

public interface Payment {

    void payment();
}

然后定义不同的支付方式的具体实现。

支付宝支付:

public class AliPayPayment implements Payment {

    @Override
    public void payment() {
        System.out.println("支付宝支付");
    }
}

微信支付:

public class WechatPayPayment implements Payment {

    @Override
    public void payment() {
        System.out.println("微信支付");
    }
}

银联云闪付:

public class UnionPayPayment implements Payment {

    @Override
    public void payment() {
        System.out.println("银联云闪付");
    }
}

再定义策略模式的容器类。

public class PaymentContext {

    private Payment payment;

    public PaymentContext(Payment payment) {
        this.payment = payment;
    }

    public void payment(){
        payment.payment();
    }
}

定义好了之后,现在再看支付的时候,我们只需要根据不同的支付方式,初始化策略容器并传入对应支付方式实现,然后调用策略容器的 payment() 方法即可。

public static void main(String[] args) {
    //支付宝支付
    PaymentContext aliPay = new PaymentContext(new AliPayPayment());
    aliPay.payment();
    //微信支付
    PaymentContext wechatPay = new PaymentContext(new WechatPayPayment());
    wechatPay.payment();
    //银联云闪付
    PaymentContext unionPay = new PaymentContext(new UnionPayPayment());
    unionPay.payment();
}

这样做了之后,虽说我们是将不同的支付方式的实现都封装到了具体的实现类中,都独立了出来,但是我们在确认初始化策略容器的时候到底应该传如那种支付方式的实现的时候,我们还是要去先判断一下支付方式,然后才能明确到底应该传入那种支付方式的实现,那么应该如何简化一下呢?

这里我们可以考虑先将所有的策略都先放到策略容器中,然后,我们再将策略容器的 payment() 方法加上一个参数,我们根据这个参数在 payment() 方法中来确定到底应该执行哪个实现。

现在进行一下改造,我们新增一个支付方式枚举:

@Getter
@AllArgsConstructor
public enum PaymentWay {

    ALI_PAY("支付宝支付"),

    WECHAT_PAY("微信支付"),

    UNION_PAY("银联云闪付支付");

    private String desc;
}

修改策略接口,新增一个获取支付方式的方法:

public interface Payment {

    void payment();

    PaymentWay getPaymentWay();
}

然后具体的支付方式的实现也加上 getPaymentWay() 方法的实现,并且返回对应的支付方式枚举。

支付宝支付:

public class AliPayPayment implements Payment {

    @Override
    public void payment() {
        System.out.println("支付宝支付");
    }

    @Override
    public PaymentWay getPaymentWay() {
       return PaymentWay.ALI_PAY;
    }
}

微信支付:

public class WechatPayPayment implements Payment {

    @Override
    public void payment() {
        System.out.println("微信支付");
    }
   
    @Override
    public PaymentWay getPaymentWay() {
        return PaymentWay.WECHAT_PAY;
    }
}

银联云闪付:

public class UnionPayPayment implements Payment {

    @Override
    public void payment() {
        System.out.println("银联云闪付");
    }

    @Override
    public PaymentWay getPaymentWay() {
       return PaymentWay.UNION_PAY;
   }
}

接着改造一下策略容器:

public class PaymentContext {

    /**
     * 将所有实现了 {@link Payment} 接口的实现都放进来
     * 具体怎么获取接口的所有实现,这里就不展开细讲了
     * 如果使用了 Spring 的话,直接使用 {@link Autowired} 注解即可将所有的实现注入进来
     */
   private List<Payment> payment;

   private Map<PaymentWay, Payment> paymentProvider;

    /**
     * 这里根据支付方式枚举为 key,以具体支付实现为 value 将所有的支付实现放到 paymentProvider 中
     */
   public PaymentContext() {
       payments.forEach(payment -> {
           paymentProvider.put(payment.getPaymentWay(), payment);
       });
   }

   public void payment(PaymentWay paymentWay) {
       paymentProvider.get(paymentWay).payment();
   }
}

改造完了之后,我们只需要初始化一次策略容器,然后调用 payment 方法的时候,直接传入支付方式,就能自动的匹配到该使用哪个支付方式的实现了。

public static void main(String[] args) {

    PaymentContext paymentContext = new PaymentContext();
    //支付宝支付
    paymentContext.payment(PaymentWay.ALI_PAY);
    //微信支付
    paymentContext.payment(PaymentWay.WECHAT_PAY);
    //银联云闪付支付
    paymentContext.payment(PaymentWay.UNION_PAY);
}

尾巴

策略模式讲完了,主要是实现的思路,如果有不严谨的地方还请见谅。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,921评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,635评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,393评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,836评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,833评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,685评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,043评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,694评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,671评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,670评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,779评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,424评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,027评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,984评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,214评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,108评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,517评论 2 343