• 设计模式18---设计模式之策略模式(Strategy)(行为型)


    1.场景模拟

    简单的报价管理系统:
    对于普通用户和新用户报全价
    对于老客户统一折扣5%
    对于大客户统一折扣10%

    2.不用模式的解决方案

    package demo16.strategy.example2;
    /**
     * 价格管理,主要完成计算向客户所报价格的功能
     */
    public class Price {
    	/**
    	 * 报价,对不同类型的,计算不同的价格
    	 * @param goodsPrice 商品销售原价
    	 * @param customerType 客户类型
    	 * @return 计算出来的,应该给客户报的价格
    	 */
    	public double quote(double goodsPrice,String customerType){
    		if("普通客户".equals(customerType)){
    			return this.calcPriceForNormal(goodsPrice);
    		}else if("老客户".equals(customerType)){
    			return this.calcPriceForOld(goodsPrice);
    		}else if("大客户".equals(customerType)){
    			return this.calcPriceForLarge(goodsPrice);		
    		}
    		//其余人员都是报原价
    		return goodsPrice;
    	}
    	/**
    	 * 为新客户或者是普通客户计算应报的价格
    	 * @param goodsPrice 商品销售原价
    	 * @return 计算出来的,应该给客户报的价格
    	 */
    	private double calcPriceForNormal(double goodsPrice){
    		System.out.println("对于新客户或者是普通客户,没有折扣");
    		return goodsPrice;
    	}
    	/**
    	 * 为老客户计算应报的价格
    	 * @param goodsPrice 商品销售原价
    	 * @return 计算出来的,应该给客户报的价格
    	 */
    	private double calcPriceForOld(double goodsPrice){
    		System.out.println("对于老客户,统一折扣5%");
    		return goodsPrice*(1-0.05);
    	}
    	/**
    	 * 为大客户计算应报的价格
    	 * @param goodsPrice 商品销售原价
    	 * @return 计算出来的,应该给客户报的价格
    	 */
    	private double calcPriceForLarge(double goodsPrice){
    		System.out.println("对于大客户,统一折扣10%");
    		return goodsPrice*(1-0.1);	
    	}
    }

    3.有何问题?

    会经常有这样的需要,在公司周年庆的时候,所有的客户额外增加3%的折扣,在换季的时候普通用户额外增加3%的折扣,然后过了促销时间,价格又要涨回来,那么这个价格类将会非常的庞大,而且,方法非常的多。
    看到这,朋友们很快就想到:如何实现才能让价格类中的计算报价的算法,很容易的实现可维护,可扩展,而且可以动态的切换变化呢?

    4.使用策略模式来解决问题

    4.1策略模式定义

    定义一系列的算法,把他们一个个封装起来,并且可以使他们相互替换,本模式使得算法可独立于使用它的客户而变化。

    4.2策略模式的结构图

     

    4.3策略模式示例代码

    package demo16.strategy.example3;
    
    
    /**
     * 策略,定义算法的接口
     */
    public interface Strategy {
    	/**
    	 * 某个算法的接口,可以有传入参数,也可以有返回值
    	 */
    	public void algorithmInterface();
    }
    *********************************************************************
    package demo16.strategy.example3;
    
    
    /**
     * 实现具体的算法
     */
    public class ConcreteStrategyA implements Strategy {
    
    
    	public void algorithmInterface() {
    		//具体的算法实现		
    	}
    	
    }
    *********************************************************************
    package demo16.strategy.example3;
    
    
    /**
     * 实现具体的算法
     */
    public class ConcreteStrategyB implements Strategy {
    
    
    	public void algorithmInterface() {
    		//具体的算法实现		
    	}
    	
    }
    
    
    package demo16.strategy.example3;
    
    
    /**
     * 实现具体的算法
     */
    public class ConcreteStrategyC implements Strategy {
    
    
    	public void algorithmInterface() {
    		//具体的算法实现		
    	}
    }
    *********************************************************************
    package demo16.strategy.example3;
    
    
    /**
     * 上下文对象,通常会持有一个具体的策略对象
     */
    public class Context {
    	/**
    	 * 持有一个具体的策略对象
    	 */
    	private Strategy strategy;
    	/**
    	 * 构造方法,传入一个具体的策略对象
    	 * @param aStrategy 具体的策略对象
    	 */
    	public Context(Strategy aStrategy) {
    		this.strategy = aStrategy;
    	}
    	/**
    	 * 上下文对客户端提供的操作接口,可以有参数和返回值
    	 */
    	public void contextInterface() {
    		//通常会转调具体的策略对象进行算法运算
    		strategy.algorithmInterface();
    	}
    }

    5.使用策略模式重写实例

    package demo16.strategy.example4;
    
    
    /**
     * 策略,定义计算报价算法的接口
     */
    public interface Strategy {
    	/**
    	 * 计算应报的价格
    	 * @param goodsPrice 商品销售原价
    	 * @return 计算出来的,应该给客户报的价格
    	 */
    	public double calcPrice(double goodsPrice);
    }
    ************************************************************************
    package demo16.strategy.example4;
    /**
     * 具体算法实现,为新客户或者是普通客户计算应报的价格
     */
    public class NormalCustomerStrategy implements Strategy{
    	public double calcPrice(double goodsPrice) {
    		System.out.println("对于新客户或者是普通客户,没有折扣");
    		return goodsPrice;
    	}
    }
    ***********************************************************************
    package demo16.strategy.example4;
    /**
     * 具体算法实现,为老客户计算应报的价格
     */
    public class OldCustomerStrategy implements Strategy{
    	public double calcPrice(double goodsPrice) {
    		System.out.println("对于老客户,统一折扣5%");
    		return goodsPrice*(1-0.05);
    	}
    }
    ***********************************************************************
    package demo16.strategy.example4;
    /**
     * 具体算法实现,为大客户计算应报的价格
     */
    public class LargeCustomerStrategy implements Strategy{
    	public double calcPrice(double goodsPrice) {
    		System.out.println("对于大客户,统一折扣10%");
    		return goodsPrice*(1-0.1);
    	}
    }
    ***********************************************************************
    package demo16.strategy.example4;
    /**
     * 价格管理,主要完成计算向客户所报价格的功能
     */
    public class Price {
    	/**
    	 * 持有一个具体的策略对象
    	 */
    	private Strategy strategy = null;
    	/**
    	 * 构造方法,传入一个具体的策略对象
    	 * @param aStrategy 具体的策略对象
    	 */
    	public Price(Strategy aStrategy){
    		this.strategy = aStrategy;
    	}	
    	/**
    	 * 报价,计算对客户的报价
    	 * @param goodsPrice 商品销售原价
    	 * @return 计算出来的,应该给客户报的价格
    	 */
    	public double quote(double goodsPrice){
    		return this.strategy.calcPrice(goodsPrice);
    	}
    }
    ***********************************************************************
    package demo16.strategy.example4;
    
    
    public class Client {
    	public static void main(String[] args) {
    		//1:选择并创建需要使用的策略对象
    		Strategy strategy = new LargeCustomerStrategy();
    		//2:创建上下文
    		Price ctx = new Price(strategy);
    		
    		//3:计算报价
    		double quote = ctx.quote(1000);
    		System.out.println("向客户报价:"+quote);
    	}
    }

    6.模式讲解

    6.1要点

    功能:把具体的算法从具体的业务处理中独立出来,把他们实现成为单独的算法类,从而形成一系列的算法,并让这些算法可以相互替换。
    策略算法是相同行为的不同实现
    什么时候选用
    :多个if-else语句的时候就可以选用

    6.2策略模式的调用顺序示意图

    首先客户端选择并创建具体的策略对象
    其次创建上下文
    最后调用上下文的方法来执行功能了
     

    6.3当添加新的策略时候

    只需要动下面代码,是不是很简单,很方便呢?
    //1:选择并创建需要使用的策略对象
    Strategy strategy = new LargeCustomerStrategy();

    6.4另一种策略模式调用示意图

     

    6.5策略模式优缺点

    优点:定义一系列算法,避免使用多重条件语句,更好的扩展性
    缺点:客户端必须了解每种策略的不同,增加了对象数目,只适合扁平的算法结构(地位平等的算法)

    6.6设计模式的本质

    分离算法,选择实现

  • 相关阅读:
    柔性数组成员 (flexible array member)-C99-ZZ
    如何阅读 Redis 源码?ZZ
    linux下网络编程学习——入门实例ZZ
    leetcode Ch2-Dynamic Programming [2014]
    leetcode Ch1-search 2014
    Skip List & Bloom Filter
    指针的引用-ZZ
    leetcode-sudoku solver
    rest framework之过滤组件
    rest framework之渲染器
  • 原文地址:https://www.cnblogs.com/jiangu66/p/3225879.html
Copyright © 2020-2023  润新知