设计模式--策略模式(由简单工厂到策略模式到两者结合图文详解+总结提升)

简介: 设计模式--策略模式(由简单工厂到策略模式到两者结合图文详解+总结提升)

概述

概念

    策略模式是一种行为型设计模式,它定义了算法家族,分别封装起来,让它们之间可以互相替换,此模式让算法的变化,不会影响到使用算法的客户。

组成

1、Context(上下文):用于维护策略类的引用,并在运行时根据需要选择合适的策略。

2、Strategy(策略):定义一个公共接口,用于封装不同的算法或行为。

3、ConcreteStrategy(具体策略):实现策略接口,提供具体的算法或行为。

应用场景

    当一个系统需要根据不同的情况选择不同的算法或行为时,可以使用策略模式。

    当一个类中有多个条件语句,每个条件语句对应一个算法或行为时,可以考虑使用策略模式简化代码。

注意事项

    策略模式将算法或行为封装在不同的策略类中,因此需要对策略类进行合理的设计和组织。

    策略模式需要在运行时动态选择策略,因此需要在客户端代码中设置策略的选择逻辑。

类图

衍化过程

需求

    商场根据不同的促销活动,以不同的方式对商品进行折扣计算和结算。具体来说,商场收银的需求可以分为以下几个阶段的变化过程:

  1. 初始需求:商场最开始只有一种收费方式,即正常收费,即不打折不返现,直接按照原价进行结算。
  2. 变化需求1:商场决定进行促销活动,对部分商品进行打折处理。这时,商场需要根据商品的折扣率来计算折扣后的价格,并进行结算。
  3. 变化需求2:商场进一步决定进行满减活动,即满足一定条件后,可以返现一定金额。这时,商场需要根据商品的价格和满减条件来计算返现后的价格,并进行结算。
  4. 变化需求3:商场希望能够根据不同的促销活动,动态地选择不同的收费方式。这时,商场引入了策略模式,将不同的收费方式封装在不同的策略类中,并在运行时根据需要选择合适的策略进行结算。

前两版比较简单,这里不再赘述,主要从需求3说起:

    分为简单工厂实现、策略模式实现、简单工厂+策略模式

简单工厂实现

代码

收费相关类

//收费抽象类
abstract class CashSuper {
    public abstract double acceptCash(double money);
}
//正常收费类
public class CashNormal extends CashSuper {
    @Override
    public double acceptCash(double money) {
        return money;
    }
}
//打折收费类
public class CashRebate extends CashSuper {
    private double moneyRebate=1;
    public CashRebate(String moneyRebate){
        this.moneyRebate= Double.parseDouble(moneyRebate);
    }
    @Override
    public double acceptCash(double money) {
        return money*moneyRebate;
    }
}
//满减收费类
public class CashReturn extends CashSuper {
    private double moneyCondtion;
    private double moneyReturn;
    public CashReturn(String moneyCondition,String moneyReturn){
        this.moneyCondtion= Double.parseDouble(moneyCondition);
        this.moneyReturn= Double.parseDouble(moneyReturn);
    }
    @Override
    public double acceptCash(double money) {
        double result=money;
        if (money>=moneyCondtion){
            result=money-Math.floor(money/moneyCondtion)*moneyReturn;
        }
        return result;
    }
}

工厂

public class CashFactory {
    public  CashSuper creatCashAccept(String type) {
        CashSuper cs = null;
        switch (type){
            case "正常收费":
                cs=new CashNormal();
            break;
            case "满300返100":
                cs=new CashReturn("300","100");
            break;
            case "打八折":
                cs=new CashRebate("0.8");
                break;
        }
        return cs;
    }

客户端

public class Client {
    public static void main(String[] args) {
        CashFactory factory=new CashFactory();
        CashSuper cs=factory.creatCashAccept("满300返100");
        double result=cs.acceptCash(300);
        System.out.println(result);
    }
}

策略模式

类图

NS图

代码

收费上下文类

public class CashContext {
    private CashSuper cs;
    public CashContext(CashSuper csuper){
        this.cs=csuper;
    }
    public double getResult(double money){
        return cs.acceptCash(money);
    }
}

客户端

public class Client {
    public static void main(String[] args) {
        //传的是不同的策略对象
        CashContext selectNormal = new CashContext(new CashNormal());
        CashContext selectRebate = new CashContext(new CashRebate("0.8"));
        CashContext selectReturn = new CashContext(new CashReturn("300","100"));
        System.out.println("选择正常:"+selectNormal.getResult(300));
        System.out.println("选择打八折:"+selectRebate.getResult(300));
        System.out.println("选择满减:"+selectReturn.getResult(300));
    }
}

策略模式+简单工厂

代码

收费上下文类(其中包括简单工厂)

public class CashContext {
   private CashSuper cs=null;
   //参数是个字符串,而不是收费策略对象
   public CashContext(String type){
       switch (type){
           case "正常收费":
               cs=new CashNormal();
               break;
           case "满300返100":
               cs=new CashReturn("300","100");
               break;
           case "打8折":
               cs=new CashRebate("0.8");
               break;
       }
   }
   public double getResult(double money){
       return cs.acceptCash(money);
   }
}

客户端

public class Client {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入销售模式:");
        String type = scanner.nextLine();
        //传的是字符串,而不是具体的策略类
        CashContext cash = new CashContext(type);
        System.out.println("请输入原价");
        double price = Double.parseDouble(scanner.nextLine());
        System.out.println("最后收费" + cash.getResult(price));
    }
}

总结升华

版本迭代的优化点及意义

1.前两版到简单工厂

     将算法封装,每一个优惠策略都是一个单独的类,而且它们都继承自一个抽象类,抽象类是此版本的一个巨大进步,出现了面向对象以及类的概念,后续的扩充、维护、复用打下了基础。

2.简单工厂到策略模式

    使用简单工厂违反开闭原则,考虑到了算法变化性,要这些算法随时可以互相替换,也正是策略模式概念中提到的,而封装变化点是面向对象一种重要的思维方式。

3.策略模式到简单工厂+策略模式

    策略模式中实例化对象还是交给了客户端去完成,结合了简单工厂之后,实例化对象由CashContext来完成,可以在客户端消除条件语句。

    另外,简单工厂模式客户端需要认识CashSuper和CashFactory,而简单工厂+策略模式,客户端之需要认识CashContext一个类就可以,降低了耦合性。

什么样的思路进行衍化的

有限到无限:

    依据业务需求,如果存在潜在的需要,就用无限的思维去进行设计(看到switch case就要思考是否需要无限的思维)

    如何去做到无限,使用想象力与创造力把有限的需求变成无限的需求,同时使用有限的方法去实现,落实到代码上是依据软件工程的7大原则和23个设计模式(独立或组合),达到可扩充、可复用、可维护的目标。

扩展思考–如何理解策略与算法

    策略模式中提到策略模式定义了算法家族,看类图的时候貌似策略和算法是一个概念,但是两者真的是一样的吗,如无必要勿增实体。那它们是什么关系呢,有没有一种可能,策略和算法并不是一一对应的关系,而独立算法和算法的组合都可以称之为策略呢,就像类图中Strategy与Context是聚合关系。这只是一个想法,策略模式也没有结束,后续还有策略模式+工厂方法+反射的实现,以及我所猜测的算法的组合也可称之为策略,敬请期待。

相关文章
|
27天前
|
设计模式 算法 Kotlin
Kotlin - 改良设计模式 - 策略模式
Kotlin - 改良设计模式 - 策略模式
45 4
|
18天前
|
设计模式 算法 Kotlin
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
36 1
|
21天前
|
设计模式 前端开发 JavaScript
JavaScript设计模式及其在实战中的应用,涵盖单例、工厂、观察者、装饰器和策略模式
本文深入探讨了JavaScript设计模式及其在实战中的应用,涵盖单例、工厂、观察者、装饰器和策略模式,结合电商网站案例,展示了设计模式如何提升代码的可维护性、扩展性和可读性,强调了其在前端开发中的重要性。
25 2
|
1月前
|
设计模式 算法 Kotlin
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
43 2
|
2月前
|
设计模式 算法 Kotlin
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
本教程详细讲解Kotlin语法,适合深入学习。快速入门可参考“简洁”系列教程。本文通过游泳运动员的案例,介绍策略模式及其在Kotlin中的改良应用,利用高阶函数简化代码结构,提高灵活性。
35 3
|
2月前
|
设计模式 算法 Kotlin
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
本教程详细讲解Kotlin语法,适合深入学习。快速入门可参考“简洁”系列教程。本文介绍策略模式在Kotlin中的应用,通过游泳运动员的例子,展示如何使用接口和高阶函数实现策略模式,使代码更简洁、灵活。
31 2
|
2月前
|
设计模式 算法 Kotlin
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
64 3
|
2月前
|
设计模式 算法 Kotlin
Kotlin - 改良设计模式 - 策略模式
Kotlin - 改良设计模式 - 策略模式
|
2月前
|
设计模式 算法 PHP
PHP中的设计模式:策略模式的深入解析与实践
【10月更文挑战第12天】 在软件开发的世界中,设计模式是解决常见问题的最佳实践。它们不是具体的代码,而是一种编码和设计经验的总结。在PHP开发中,合理运用设计模式可以极大地提高代码的可维护性、扩展性和复用性。本文将深入探讨策略模式(Strategy Pattern)的原理、实现方式及其在PHP中的应用。通过具体示例,我们将展示如何利用策略模式来解耦算法与对象,从而让代码更加灵活和易于管理。
19 0
|
2月前
|
设计模式 存储 算法
PHP中的设计模式:策略模式的深入解析与实践
【10月更文挑战第9天】 在PHP开发领域,设计模式是提升代码可维护性、扩展性和重用性的关键技术之一。本文聚焦于策略模式这一行为型设计模式,通过理论阐述与实例分析,揭示其在PHP应用程序中优化算法切换和业务逻辑解耦方面的强大效用。不同于常规摘要,本文不直接概述研究方法或结果,而是基于实际开发场景,探讨策略模式的应用价值和实现方式,旨在为PHP开发者提供一种高效应对复杂业务需求变化和技术债务累积问题的策略思维。

热门文章

最新文章

  • 1
    设计模式转型:从传统同步到Python协程异步编程的实践与思考
    58
  • 2
    C++一分钟之-设计模式:工厂模式与抽象工厂
    47
  • 3
    《手把手教你》系列基础篇(九十四)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-下篇(详解教程)
    54
  • 4
    C++一分钟之-C++中的设计模式:单例模式
    65
  • 5
    《手把手教你》系列基础篇(九十三)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-上篇(详解教程)
    43
  • 6
    《手把手教你》系列基础篇(九十二)-java+ selenium自动化测试-框架设计基础-POM设计模式简介(详解教程)
    70
  • 7
    Java面试题:结合设计模式与并发工具包实现高效缓存;多线程与内存管理优化实践;并发框架与设计模式在复杂系统中的应用
    62
  • 8
    Java面试题:设计模式在并发编程中的创新应用,Java内存管理与多线程工具类的综合应用,Java并发工具包与并发框架的创新应用
    43
  • 9
    Java面试题:如何使用设计模式优化多线程环境下的资源管理?Java内存模型与并发工具类的协同工作,描述ForkJoinPool的工作机制,并解释其在并行计算中的优势。如何根据任务特性调整线程池参数
    52
  • 10
    Java面试题:请列举三种常用的设计模式,并分别给出在Java中的应用场景?请分析Java内存管理中的主要问题,并提出相应的优化策略?请简述Java多线程编程中的常见问题,并给出解决方案
    121