策略(Strategy)模式:又名Policy,它的用意是定义一组算法,把它们一个个封装起来,并且使他们可以相互替换。策略模式可以独立于使用他们的客户端而变化。GOF策略模式静态结构类图如下:
通过上图可以看出策略模式有以下角色构成:
1、抽象策略(Strategy)角色:抽象策略角色由抽象类或接口来承担,它给出具体策略角色需要实现的接口;
2、具体策略(ConcreteStrategy)角色:实现封装了具体的算法或行为;
3、场景(Context)角色:持有抽象策略类的引用。
策略模式重点是封装不同的算法和行为,不同的场景下可以相互替换。策略模式是开闭原则的体现,开闭原则讲的是一个软件实体应该对扩展开放对修改关闭。策略模式在新的策略增加时,不会影响其他类的修改,增加了扩展性,也就是对扩展是开放的;对于场景来说,只依赖于抽象,而不依赖于具体实现,所以对修改是关闭的。策略模式的认识可以借助《java与模式》一书中写到诸葛亮的锦囊妙计来学习,在不同的场景下赵云打开不同的锦囊,便化险为夷,锦囊便是抽象策略,具体的锦囊里面的计策便是具体的策略角色,场景就是赵云,变化的处境
选择具体策略的条件。
策略模式在程序设计中也很常用,在板桥(banq)的博客里有篇文章叫 “你还在用if else吗?”
“http://www.jdon.com/artichect/ifelse.htm”讲的很好,策略模式不但是继承的代替方案而且能很好地解决if else问题,下面举个实例来说明,怎么使用策略模式。
需求如下:
某支付系统接入以下几种商户进行充值:易宝网易,快线网银,19pay手机支付,支付宝支付,骏网一卡通,由于每家充值系统的结算比例不一样,而且同一家商户的不同充值方式也有所不同,具体系统情况比较复杂,像支付宝既有支付宝账号支付和支付宝网银支付等这些暂时不考虑,为了讲述策略模式这里简单描述,假如分为四种,手机支付,网银支付,商户账号支付和点卡支付。因为没个支付结算比例不同,所以对手续费低的做一些优惠活动,尽可能让用户使用手续费低的支付方式来充值,这样降低渠道费用,增加收入,具体优惠政策如下:
①网银充值,8.5折;
②商户充值,9折;
③手机充值,没有优惠;
④点卡充值,收取1%的渠道费;
对于一个新手的代码如下:
package strategy; public class Example { /** * *作者:alaric *时间:2013-8-5上午11:00:06 *描述:计算用户所付金额 */ public Double calRecharge(Double charge ,RechargeTypeEnum type ){ if(type.equals(RechargeTypeEnum.E_BANK)){ return charge*0.85; }else if(type.equals(RechargeTypeEnum.BUSI_ACCOUNTS)){ return charge*0.90; }else if(type.equals(RechargeTypeEnum.MOBILE)){ return charge; }else if(type.equals(RechargeTypeEnum.CARD_RECHARGE)){ return charge+charge*0.01; }else{ return null; } } }
package strategy; public enum RechargeTypeEnum { E_BANK(1, "网银"), BUSI_ACCOUNTS(2, "商户账号"), MOBILE(3,"手机卡充值"), CARD_RECHARGE(4,"充值卡") ; /** * 状态值 */ private int value; /** * 类型描述 */ private String description; private RechargeTypeEnum(int value, String description) { this.value = value; this.description = description; } public int value() { return value; } public String description() { return description; } public static RechargeTypeEnum valueOf(int value) { for(RechargeTypeEnum type : RechargeTypeEnum.values()) { if(type.value() == value) { return type; } } return null; } }
可以看出上面四种不同的计算方式在一个方法内部,不利于扩展和维护,当然也不符合面向对象设计原则。对以上的代码利用策略模式进行修改,类图如下:
实例代码如下:
package strategy.strategy; import strategy.RechargeTypeEnum; /** * *作者:alaric *时间:2013-8-5上午11:03:17 *描述:策略抽象类 */ public interface Strategy { /** * *作者:alaric *时间:2013-8-5上午11:05:11 *描述:策略行为方法 */ public Double calRecharge(Double charge ,RechargeTypeEnum type ); }
package strategy.strategy; import strategy.RechargeTypeEnum; /** * *作者:alaric *时间:2013-8-5上午11:14:23 *描述:网银充值 */ public class EBankStrategy implements Strategy{ @Override public Double calRecharge(Double charge, RechargeTypeEnum type) { return charge*0.85; } }
package strategy.strategy; import strategy.RechargeTypeEnum; /** * *作者:alaric *时间:2013-8-5上午11:14:08 *描述:商户账号充值 */ public class BusiAcctStrategy implements Strategy{ @Override public Double calRecharge(Double charge, RechargeTypeEnum type) { // TODO Auto-generated method stub return charge*0.90; } }
package strategy.strategy; import strategy.RechargeTypeEnum; /** * *作者:alaric *时间:2013-8-5上午11:14:43 *描述:手机充值 */ public class MobileStrategy implements Strategy { @Override public Double calRecharge(Double charge, RechargeTypeEnum type) { // TODO Auto-generated method stub return charge; } }
package strategy.strategy; import strategy.RechargeTypeEnum; /** * *作者:alaric *时间:2013-8-5上午11:13:46 *描述:充值卡充值 */ public class CardStrategy implements Strategy{ @Override public Double calRecharge(Double charge, RechargeTypeEnum type) { return charge+charge*0.01; } }
package strategy.strategy; import strategy.RechargeTypeEnum; /** * *作者:alaric *时间:2013-8-5上午11:03:38 *描述:场景类 */ public class Context { private Strategy strategy; public Double calRecharge(Double charge, Integer type) { strategy = StrategyFactory.getInstance().creator(type); return strategy.calRecharge(charge, RechargeTypeEnum.valueOf(type)); } public Strategy getStrategy() { return strategy; } public void setStrategy(Strategy strategy) { this.strategy = strategy; } }
package strategy.strategy; import java.util.HashMap; import java.util.Map; import strategy.RechargeTypeEnum; /** * *作者:alaric *时间:2013-8-5上午11:31:12 *描述:策略工厂 使用单例模式 */ public class StrategyFactory { private static StrategyFactory factory = new StrategyFactory(); private StrategyFactory(){ } private static Map<Integer ,Strategy> strategyMap = new HashMap<>(); static{ strategyMap.put(RechargeTypeEnum.E_BANK.value(), new EBankStrategy()); strategyMap.put(RechargeTypeEnum.BUSI_ACCOUNTS.value(), new BusiAcctStrategy()); strategyMap.put(RechargeTypeEnum.MOBILE.value(), new MobileStrategy()); strategyMap.put(RechargeTypeEnum.CARD_RECHARGE.value(), new CardStrategy()); } public Strategy creator(Integer type){ return strategyMap.get(type); } public static StrategyFactory getInstance(){ return factory; } }
package strategy.strategy; import strategy.RechargeTypeEnum; public class Client { /** * 作者:alaric 时间:2013-8-5上午11:33:52 描述: */ public static void main(String[] args) { Context context = new Context(); // 网银充值100 需要付多少 Double money = context.calRecharge(100D, RechargeTypeEnum.E_BANK.value()); System.out.println(money); // 商户账户充值100 需要付多少 Double money2 = context.calRecharge(100D, RechargeTypeEnum.BUSI_ACCOUNTS.value()); System.out.println(money2); // 手机充值100 需要付多少 Double money3 = context.calRecharge(100D, RechargeTypeEnum.MOBILE.value()); System.out.println(money3); // 充值卡充值100 需要付多少 Double money4 = context.calRecharge(100D, RechargeTypeEnum.CARD_RECHARGE.value()); System.out.println(money4); } }
运行结果:
85.0
90.0
100.0
101.0
设计模式系列目录:
相关推荐
浅析Java设计模式【4】——策略.pdf
NULL 博文链接:https://lvwenwen.iteye.com/blog/1456554
Java设计模式教程——策略模式Java开发Java经验技巧共13页.pdf.zip
Java设计模式,目录:前言,UML建模技术,深入浅出UML类图,从招式与内功谈起——设计模式概述,面向对象设计原则,工厂三兄弟之简单工厂模式,工厂三兄弟之工厂方法模式,工厂三兄弟之抽象工厂模式,确保对象的唯一...
NULL 博文链接:https://lvwenwen.iteye.com/blog/1549415
目录: 前 言 第一部分 大旗不挥,谁敢冲锋——热身篇 第1章 单一职责原则 1.1 我是“牛”类,我可以担任多职吗 1.2 绝杀技,打破你的传统思维 1.3 我单纯,所以我快乐 1.4 最佳实践 ...附录:23个设计模式
4.4.2现实中的装饰模式——相架 126 4.4.3C#实例——图书馆中的项目 127 4.4.4Java实例——自定义JButton 131 4.4.5优势和缺陷 133 4.4.6应用情景 134 4.5FacadePattern(外观模式) 134 4.5.1定义 134 4.5.2...
4.4.2现实中的装饰模式——相架 126 4.4.3C#实例——图书馆中的项目 127 4.4.4Java实例——自定义JButton 131 4.4.5优势和缺陷 133 4.4.6应用情景 134 4.5FacadePattern(外观模式) 134 4.5.1定义 134 4.5.2...
java和设计模式ppt包含工厂模式、建造模式、原始模型模式、单例模式、结构模式、适配器、桥梁模式、合成模式、装饰模式、门面模式、享元模式、代理模式、行为模式、解释器模式、迭代子模式、调停者模式、备忘录模式...
设计模式精解(Design Patterns Explained) ...如何实现关键模式——Strategy(策略)、Observer(观察者)、Bridge(桥接)、Decorator(装饰)等等。 共同点/变化点分析、设计模式以及它们如何帮助理解抽象类。
设计模式趣味学习(复习) 设计模式趣味学习(复习) 设计模式与足球(一) 设计模式与足球(二) 设计模式与足球(三) 设计模式与足球(四) 设计模式综合应用实例 设计模式综合应用实例 多人联机射击游戏 多人...
策略模式工作坊陈述我们希望扩展在之前的研讨会中开发的保存策略(JSON 和 XML)的功能,添加缓存、加密 (AES) 和操作日志记录服务。 这些服务是可选的,客户可能希望组合这些服务,无论使用哪种保存策略,这些服务...
综合实例——Bug管理系统 (3)设计模式样例(24个讲解样例程序) pattern/src/principle/liskovsubstitution//10.3.2里氏代换原则 pattern/src/creation/factorymethod //11.1工厂方法模式 pattern/src/creation/...
本文以项目中的一个工作流模块,演示责任链模式、策略模式、命令模式的组合实现!最近在做的一个项目,涉及到的是一个流程性质的需求。关于工程机械行业的服务流程:服务任务流程和备件发运流程。项目之初,需求不是...
java设计模式期末大作业,运用了6种模式,包括简单工厂模式、工厂方法模式、单例模式、门面模式、策略模式、观察者模式,文档包括系统流程,系统类图,各个模式的子类图,源代码,实验截图。绝对完整.
综合实例——Bug管理系统 (3)设计模式样例(24个讲解样例程序) pattern/src/principle/liskovsubstitution//10.3.2里氏代换原则 pattern/src/creation/factorymethod //11.1工厂方法模式 pattern/src/creation/...
pattern/src/behavior/strategy //13.1策略模式 pattern/src/behavior/templatemethod //13.2模板方法模式 pattern/src/behavior/observer //13.3观察者模式 pattern/src/behavior/iterator //13.4迭代子模式 ...
避免使用Java序列化14讲多线程之锁优化(下):使用乐观锁优化并行操作16讲多线程调优(下):如何优化多线程上下文切换17讲并发容器的使用:识别不同场景下最优容器21讲深入JVM即时编译器JIT,优化Java编译25讲答疑...
pattern/src/behavior/strategy //13.1策略模式 pattern/src/behavior/templatemethod //13.2模板方法模式 pattern/src/behavior/observer //13.3观察者模式 pattern/src/behavior/iterator //13.4迭代子模式 ...
综合实例——Bug管理系统 (3)设计模式样例(24个讲解样例程序) pattern/src/principle/liskovsubstitution//10.3.2里氏代换原则 pattern/src/creation/factorymethod //11.1工厂方法模式 pattern/src/creation/...