热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

策略模式–设计模式之行为模式

策略模式–设计模式之行为模式目录定义:类图:例子:类图:抽象的策略角色:IDiscount具体策

策略模式 – 设计模式之行为模式

目录

定义:

类图:

例子:

类图:

抽象的策略角色: IDiscount

具体策略角色1 OrdinaryStrategy

具体策略角色2 PlatinumStrategy

具体策略角色3 GoldStrategy

封装角色  DiscountStrategyContext

测试; DiscountTest

优化1:结合工厂方法

枚举类: UserTypeEnum

抽象的策略角色: IDiscount

具体策略角色1 OrdinaryStrategy

具体策略角色2 PlatinumStrategy

具体策略角色3 GoldStrategy

工厂方法类  DiscountStrategyFactory

测试; DiscountFactoryTest

优化2: 使用静态工厂,动态获取类

动态实例化类: SpringUtils

抽象的策略角色: IDiscount

具体策略角色1 OrdinaryStrategy

具体策略角色2 PlatinumStrategy

具体策略角色3 GoldStrategy

工厂方法类  DiscountStrategyFactory

封装角色; DiscountStrategyServiceContext

测试; DisountApplicationTests

总结:



 


定义:

 Define a family of algorithms, encapsulate each one, and make them interchangeable.(定义一组算法,将每个算法都封装起来,并且使它们之间可以互换)

定义了算法家族,分别封装起来,让它们之间可以相互替换,此模式算法的变化,不会影响到使用算法的用户。

 


类图:

 

 

 


例子:

用常见的购物场景,普通的不打折,白金会员打8折,铂金会员打6折

 


类图:

 


抽象的策略角色: IDiscount

public interface IDiscount {double compute(double money);
}

 


具体策略角色1 OrdinaryStrategy

public class OrdinaryStrategy implements IDiscount {@Overridepublic double compute(double money) {System.out.println("普通会员不打折");System.out.println("积分是1%");return money;}}

 


具体策略角色2 PlatinumStrategy

public class PlatinumStrategy implements IDiscount {@Overridepublic double compute(double money) {System.out.println("白金会员 优惠50元,再打7折");System.out.println("积分是5%");return (money - 50) * 0.7;}}

 


具体策略角色3 GoldStrategy

public class GoldStrategy implements IDiscount {@Overridepublic double compute(double money) {System.out.println("黄金会员 优惠50元,再打6折");System.out.println("积分是10%");return (money - 50)*0.6;}}

 


封装角色  DiscountStrategyContext

public  class DiscountStrategyContext {private IDiscount discountStrategy;/*** 设置策略接口* @param discountHandleStrategy*/public void setDiscountHandleStrategy(IDiscount discountHandleStrategy) {this.discountStrategy = discountHandleStrategy;}public void computeMoney(double money) {if (discountStrategy != null) {double compute = discountStrategy.compute(money);System.out.println(" pay "+ compute);}}}

 


测试; DiscountTest

public class DiscountTest {public static void main(String[] args) {IDiscount discountStrategy = new OrdinaryStrategy();DiscountStrategyContext discountStrategyCOntext= new DiscountStrategyContext();discountStrategyContext.setDiscountHandleStrategy(discountStrategy);discountStrategyContext.computeMoney(100);}}

 

结果:

普通会员不打折积分是1%pay 100.0

思考: 在使用策略模式的时候,得先知道是哪个策略,还不是得if-else 进行判断,具体使用哪个策略。这个要怎么进行优化呢?只要知道一个名字就可以了,传递相关的数字或是名字进来,反馈一个结果,这才是想要的。

传递名字: 考虑用枚举,如例子中,三种会员的类型,相对固定。

对各个策略进行集中处理,考虑用工厂方法模式来实现策略类的声明。要知道策略的名称,接口那需要添加一个获取类型的方法。

 


优化1:结合工厂方法

 


枚举类: UserTypeEnum

public enum UserTypeEnum {COMMON_USER("COMMON"), PLATINUM_VIP("PLATINUM"),  GOLD_VIP("GOLD");String type;public String getType() {return type;}UserTypeEnum(String type) {this.type = type;}public static UserTypeEnum matchType(String type) {for (UserTypeEnum enumType : UserTypeEnum.values()) {if (enumType.type == type) {return enumType;}}return UserTypeEnum.COMMON_USER;}}

 


抽象的策略角色: IDiscount

public interface IDiscount {double compute(double money);// 用于匹配类型String getType();}

 


具体策略角色1 OrdinaryStrategy

public class OrdinaryStrategy implements IDiscount { @Overridepublic double compute(double money) {System.out.println("普通会员不打折");System.out.println("积分是1%");return money;}@Overridepublic String getType() {return UserTypeEnum.COMMON_USER.getType();}}

 


具体策略角色2 PlatinumStrategy

public class PlatinumStrategy implements IDiscount {@Overridepublic double compute(double money) {System.out.println("白金会员 优惠50元,再打7折");System.out.println("积分是5%");return (money - 50) * 0.7;}@Overridepublic String getType() {return UserTypeEnum.PLATINUM_VIP.getType();}}

 


具体策略角色3 GoldStrategy

public class GoldStrategy implements IDiscount {@Overridepublic double compute(double money) {System.out.println("黄金会员 优惠50元,再打6折");System.out.println("积分是10%");return (money - 50)*0.6;}@Overridepublic String getType() {return UserTypeEnum.GOLD_VIP.getType();}}

 

 


工厂方法类  DiscountStrategyFactory

public class DiscountStrategyFactory {private Map map;private DiscountStrategyFactory() {List strategies = new ArrayList<>();strategies.add(new OrdinaryStrategy());strategies.add(new GoldStrategy());strategies.add(new GoldStrategy());strategies.add(new PlatinumStrategy());// getType 主要在这边声明的时候用map = strategies.stream().collect(Collectors.toMap(IDiscount::getType, strategy -> strategy, (x, y) -> x));}public static class Holder {private static DiscountStrategyFactory instance = new DiscountStrategyFactory();}public static DiscountStrategyFactory getInstance() {return Holder.instance;}public IDiscount getIDiscountByType(String type) {return map.get(type);}}

 

或是用静态的方式,直接map,这样就不用声明getType() 和实现这个方法了

 DiscountStrategyFactoryStatic:

public class DiscountStrategyFactoryStatic {public static Map DICOUNT_TYPE = new HashMap<>();static {DICOUNT_TYPE.put("COMMON",new OrdinaryStrategy());DICOUNT_TYPE.put("PLATINUM",new GoldStrategy());DICOUNT_TYPE.put("GOLD",new PlatinumStrategy());}public static class Holder {private static DiscountStrategyFactoryStatic instance = new DiscountStrategyFactoryStatic();}public static DiscountStrategyFactoryStatic getInstance() {return Holder.instance;}public IDiscount getIDiscountByType(String type) {return DICOUNT_TYPE.get(type);}
}

测试; DiscountFactoryTest

public class DiscountFactoryTest {private static double getResult(long money, String type) {if (money <1000) {return money;}IDiscount strategy = DiscountStrategyFactory.getInstance().getIDiscountByType (type);if (strategy == null) {throw new IllegalArgumentException("please input right type");}return strategy.compute(money);}public static void main(String[] args) {String type = "GOLD";UserTypeEnum userTypeEnum = UserTypeEnum.matchType(type);System.out.println(getResult(3300, userTypeEnum.getType()));System.out.println(getResult(2300, "PLATINUM"));}}

 

结果:

黄金会员 优惠50元,再打6折积分是10%1950.0白金会员 优惠50元,再打7折积分是5%1575.0

 

这就达到了想要的要求。输入一个名字,一个价格,就可以得到结果。

 


优化2: 使用静态工厂,动态获取类

 


动态实例化类: SpringUtils

@Componentpublic class SpringUtils implements ApplicationContextAware {private static ApplicationContext applicationContext;@Overridepublic void setApplicationContext(ApplicationContext applicationContext) throws BeansException {SpringUtils.applicatiOnContext= applicationContext;}public static Object getBean(String name) {return applicationContext.getBean(name);}public static T getBean(Class clazz) {return applicationContext.getBean(clazz);}}

 


抽象的策略角色: IDiscount

public interface IDiscount {double compute(double money);
}

 


具体策略角色1 OrdinaryStrategy

@Service
public class OrdinaryStrategy implements IDiscount {@Overridepublic double compute(double money) {System.out.println("普通会员不打折");System.out.println("积分是1%");return money;}}

 


具体策略角色2 PlatinumStrategy

@Service
public class PlatinumStrategy implements IDiscount {@Overridepublic double compute(double money) {System.out.println("白金会员 优惠50元,再打7折");System.out.println("积分是5%");return (money - 50) * 0.7;}}

 


具体策略角色3 GoldStrategy

@Service
public class GoldStrategy implements IDiscount {@Overridepublic double compute(double money) {System.out.println("黄金会员 优惠50元,再打6折");System.out.println("积分是10%");return (money - 50)*0.6;}}

 


工厂方法类  DiscountStrategyFactory

@Servicepublic class DiscountStrategyServiceFactory {private static Map DICOUNT_TYPE = new HashMap<>();static {DICOUNT_TYPE.put("COMMON","goldStrategyService");DICOUNT_TYPE.put("PLATINUM","platinumStrategyService");DICOUNT_TYPE.put("GOLD","goldStrategyService");}public IDiscount getDiscountService(String type){return (IDiscount)SpringUtils.getBean(DICOUNT_TYPE.get(type));}}

 

这边主要用SpringUtils获取bean的方式,实例化type对应的类

 


封装角色; DiscountStrategyServiceContext

@Servicepublic class DiscountStrategyServiceContext {@Resourceprivate  DiscountStrategyServiceFactory discountStrategyServiceFactory;private double getResult(long money, String type) {if (money <1000) {return money;}IDiscount strategy = discountStrategyServiceFactory.getDiscountService(type);if (strategy == null) {throw new IllegalArgumentException("please input right type");}return strategy.compute(money);}public void testResult() {System.out.println(getResult(3300, "GOLD"));System.out.println(getResult(2300, "PLATINUM"));}}

 

 

 


测试; DisountApplicationTests

@RunWith(SpringRunner.class)@SpringBootTest
public class DisountApplicationTests {@Autowiredprivate DiscountStrategyServiceContext discountStrategyServiceContext;@Testpublic void testResult() {discountStrategyServiceContext.testResult();}}

PS:这个是在springBoot项目里面的,启动后 注入@Service,初始化实例

结果:

黄金会员 优惠50元,再打6折积分是10%1950.0白金会员 优惠50元,再打7折积分是5%1575.0

 


总结:

单纯用策略模式,并不能避免使用多重条件判断,结合枚举和工厂方法模式可以解决 问题,只通过一个名字或是数字,匹配到对应的策略。

策略模式的重点就是封装角色,它是借用了代理模式的思路,大家可以想想,它和代理模式有什么差别,差别就是策略模式的封装角色和被封装的策略类不用是同一个接口,如果是同一个接口那就成为了代理模式。

 


推荐阅读
  • 我在尝试将组合框转换为具有自动完成功能时遇到了一个问题,即页面上的列表框也被转换成了自动完成下拉框,而不是保持原有的多选列表框形式。 ... [详细]
  • 本文详细介绍了在PHP中如何获取和处理HTTP头部信息,包括通过cURL获取请求头信息、使用header函数发送响应头以及获取客户端HTTP头部的方法。同时,还探讨了PHP中$_SERVER变量的使用,以获取客户端和服务器的相关信息。 ... [详细]
  • 本文详细介绍了Socket在Linux内核中的实现机制,包括基本的Socket结构、协议操作集以及不同协议下的具体实现。通过这些内容,读者可以更好地理解Socket的工作原理。 ... [详细]
  • HDU 2537 键盘输入处理
    题目描述了一个名叫Pirates的男孩想要开发一款键盘输入软件,遇到了大小写字母判断的问题。本文提供了该问题的解决方案及实现方法。 ... [详细]
  • 本文将详细介绍如何配置并整合MVP架构、Retrofit网络请求库、Dagger2依赖注入框架以及RxAndroid响应式编程库,构建高效、模块化的Android应用。 ... [详细]
  • Kubernetes Services详解
    本文深入探讨了Kubernetes中的服务(Services)概念,解释了如何通过Services实现Pods之间的稳定通信,以及如何管理没有选择器的服务。 ... [详细]
  • 本文探讨了Android系统中联系人数据库的设计,特别是AbstractContactsProvider类的作用与实现。文章提供了对源代码的详细分析,并解释了该类如何支持跨数据库操作及事务处理。源代码可从官方Android网站下载。 ... [详细]
  • 使用jQuery与百度地图API实现地址转经纬度功能
    本文详细介绍了如何利用jQuery和百度地图API将地址转换为经纬度,包括申请API密钥、页面构建及核心代码实现。 ... [详细]
  • 本文介绍如何通过Java代码调用阿里云短信服务API来实现短信验证码的发送功能,包括必要的依赖添加和关键代码示例。 ... [详细]
  • 使用 ModelAttribute 实现页面数据自动填充
    本文介绍了如何利用 Spring MVC 中的 ModelAttribute 注解,在页面跳转后自动填充表单数据。主要探讨了两种实现方法及其背后的原理。 ... [详细]
  • 探索CNN的可视化技术
    神经网络的可视化在理论学习与实践应用中扮演着至关重要的角色。本文深入探讨了三种有效的CNN(卷积神经网络)可视化方法,旨在帮助读者更好地理解和优化模型。 ... [详细]
  • UVa 11683: 激光雕刻技术解析
    自1958年发明以来,激光技术已在众多领域得到广泛应用,包括电子设备、医疗手术工具、武器等。本文将探讨如何使用激光技术进行材料雕刻,并通过编程解决一个具体的激光雕刻问题。 ... [详细]
  • 本文详细介绍了如何在PHP中使用Memcached进行数据缓存,包括服务器连接、数据操作、高级功能等。 ... [详细]
  • 本文探讨了如何选择一个合适的序列化版本ID(serialVersionUID),包括使用生成器还是简单的整数,以及在不同情况下应如何处理序列化版本ID。 ... [详细]
  • STM32代码编写STM32端不需要写关于连接MQTT服务器的代码,连接的工作交给ESP8266来做,STM32只需要通过串口接收和发送数据,间接的与服务器交互。串口三配置串口一已 ... [详细]
author-avatar
lakensei
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有