09、策略模式

2019独角兽企业重金招聘Python工程师标准>>> hot3.png

策略模式与工厂模式最大的区别在于,策略模式注重的是对算法的维护,也可以理解为对算法的封装。而工厂模式,则只是负责创建类,在刚接触策略模式时候,往往与工厂模式容易产生混淆。
官方对策略模式的解释:
它定义了算法家族,分别封装起来,让他们之间可以互换,此模式,让算法的变化不会影响到使用算法的客户。

从此模式开始,每次总结的时候画出一个UML图。

此例子结合商场搞活动销售活动,商品价格打折、以及返价活动。
任何一种活动,最终都要有一个计算结果的方法,所以对此方法进行抽象:
   
  1. public interface PriceCalculate {
  2. public abstract double getFinalPrice();
  3. }
然后对其进行实现,分别为不搞活动,折扣,返现活动,分别如下:
   
  1. public class NormalPrice implements PriceCalculate {
  2. private double finalPrice = 0d;
  3. public double getFinalPrice() {
  4. // TODO Auto-generated method stub
  5. return finalPrice;
  6. }
  7. public NormalPrice(double sourcePrice){
  8. this.finalPrice = sourcePrice;
  9. }
  10. }
然后是折扣活动:
   
  1. public class DebetPrice implements PriceCalculate {
  2. private double finalPrice = 0d;
  3. @Override
  4. public double getFinalPrice() {
  5. return finalPrice;
  6. }
  7. public DebetPrice(double sourcePrice,double debet){
  8. this.finalPrice = Calculate.mul(sourcePrice, debet);
  9. }
  10. }
Calculate为工具类,具体代码在最后。
然后是折现活动:
    
  1. public class ReturnPrice implements PriceCalculate {
  2. private double finalPrice = 0d;
  3. @Override
  4. public double getFinalPrice() {
  5. // TODO Auto-generated method stub
  6. return finalPrice;
  7. }
  8. /**满多少价反钱活动
  9. * @param sourcePrice 原价
  10. * @param conditionPrice 返现金消费额度
  11. * @param returnPrice 返现金力度
  12. */
  13. public ReturnPrice(double sourcePrice,double conditionPrice,double returnPrice){
  14. this.finalPrice = sourcePrice;
  15. if( sourcePrice >= conditionPrice ){
  16. double times = Math.floor(Calculate.div(sourcePrice, conditionPrice)) ;
  17. int count = times>=1? (int)times:0;
  18. while(count-->0){
  19. this.finalPrice = Calculate.sub(sourcePrice,returnPrice);
  20. }
  21. }
  22. }
  23. }


其中的Calculate为工具类,具体如下:
   
  1. package com.yp.learn.util;
  2. import java.math.BigDecimal;
  3. public class Calculate {
  4. // 默认除法运算精度
  5. private static final int DEF_DIV_SCALE = 10;
  6. private Calculate(){};
  7. /**精确 加法运算
  8. * @param d1 被加数
  9. * @param d2 加数
  10. * @return 和
  11. */
  12. public static double add(double d1,double d2){
  13. BigDecimal b1= new BigDecimal(Double.toString(d1));
  14. BigDecimal b2= new BigDecimal(Double.toString(d2));
  15. return b1.add(b2).doubleValue();
  16. }
  17. /**
  18. * 提供精确的减法运算。
  19. * @param v1 被减数
  20. * @param v2 减数
  21. * @return 两个参数的差
  22. */
  23. public static double sub(double v1, double v2) {
  24. BigDecimal b1 = new BigDecimal(Double.toString(v1));
  25. BigDecimal b2 = new BigDecimal(Double.toString(v2));
  26. return b1.subtract(b2).doubleValue();
  27. }
  28. /**
  29. * 提供精确的乘法运算。
  30. * @param v1 被乘数
  31. * @param v2 乘数
  32. * @return 两个参数的积
  33. */
  34. public static double mul(double v1, double v2) {
  35. BigDecimal b1 = new BigDecimal(Double.toString(v1));
  36. BigDecimal b2 = new BigDecimal(Double.toString(v2));
  37. return b1.multiply(b2).doubleValue();
  38. }
  39. /**
  40. * 提供(相对)精确的除法运算,当发生除不尽的情况时,精确到 小数点以后10位,以后的数字四舍五入。
  41. * @param v1 被除数
  42. * @param v2 除数
  43. * @return 两个参数的商
  44. */
  45. public static double div(double v1, double v2) {
  46. return div(v1, v2, DEF_DIV_SCALE);
  47. }
  48. /**
  49. * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指 定精度,以后的数字四舍五入。
  50. * @param v1 被除数
  51. * @param v2 除数
  52. * @param scale 表示表示需要精确到小数点以后几位。
  53. * @return 两个参数的商
  54. */
  55. public static double div(double v1, double v2, int scale) {
  56. if (scale < 0) {
  57. throw new IllegalArgumentException(
  58. "The scale must be a positive integer or zero");
  59. }
  60. BigDecimal b1 = new BigDecimal(Double.toString(v1));
  61. BigDecimal b2 = new BigDecimal(Double.toString(v2));
  62. return b1.divide(b2, scale, BigDecimal.ROUND_HALF_UP).doubleValue();
  63. }
  64. }
然后策略模式的核心出来了,几种类都已经实现类,并且都有算法,策略模式要做的,就是 对这个抽象出来的 算法进行 维护,如下:
   
  1. public class PriceCalculateContext {
  2. PriceCalculate cal = null;
  3. /**无活动情况
  4. * @param sourcePrice
  5. */
  6. public PriceCalculateContext(double sourcePrice){
  7. cal = new NormalPrice(sourcePrice);
  8. }
  9. /**打折活动
  10. * @param sourcePrice 原价
  11. * @param debet 打折力度
  12. */
  13. public PriceCalculateContext(double sourcePrice,double debet){
  14. cal = new DebetPrice(sourcePrice,debet);
  15. }
  16. /**返现活动
  17. * @param sourcePrice 原价
  18. * @param conditionPrice 返现所需满足额度
  19. * @param returnPrice 返现力度
  20. */
  21. public PriceCalculateContext(double sourcePrice,double conditionPrice,double returnPrice){
  22. cal = new ReturnPrice(sourcePrice,conditionPrice,returnPrice);
  23. }
  24. public double getResult() {
  25. return cal.getFinalPrice();
  26. }
  27. }
与之前的工厂模式相比较,策略模式并没有提供创建运算类的工厂类,而只是将这些运算类维护在了PriceCalculateContext类中,并且此类中提供了方法getResult(),此方法所调用的刚好是算法接口。那么现在,客户端的代码就非常的清晰了:
   
  1. public class Start {
  2. public static void main(String[] args) {
  3. PriceCalculateContext p = new PriceCalculateContext(100d);
  4. System.out.println(p.getResult());
  5. p = new PriceCalculateContext(122.2d, 100d, 15d);
  6. System.out.println(p.getResult());
  7. }
  8. }
通过构造价格计算维护类来计算所得的最终结果,计算的具体形式完全对客户端屏蔽。策略模式最明显的就是对每种策略所实现的算法进行了封装,在本例中,就是 PriceCalculateContext的getResult的方法,在后续,可以说如果需要新增新的算法,那么我只需要新增新的算法类,然后在PriceCalculateContext增加相应的映射即可。在客户端只需要依据不同的参数进行初始化,然后调用的方法不用发生任何改变。
UML图如下:

从类图上也可以看到,客户端的代码只关联PriceCalculateContext类,解耦很成功。

还是强调一点,策略模式注重对策略的封装。



总结分析:
1、比较工厂模式,此模式通过增加算法维护类(PriceCalculateContext),对所有的算法进行封装,并且在客户端代码中,只有此类创建和使用。
2、当新增类的时候,需要算法维护类中修改代码,相当于需求和实现做映射关系。即客户端代码依然没有转移条件判断。




来自为知笔记(Wiz)


转载于:https://my.oschina.net/u/1182369/blog/406571

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/387575.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java编写代理服务器(Burp拦截Demo)一

大家都知道大名鼎鼎的BurpSuite代理神器&#xff0c;对于抓取HTTP请求非常好用&#xff0c;偶然&#xff0c;一朋友问我Java应该如何去编写代理服务器&#xff08;因为他想做某些东西&#xff09;&#xff0c;有没有相关的API 去实现&#xff0c;我想说&#xff0c;差不多你能想…

Rabbit MQ windows下安装

Rabbit MQ 是建立在强大的Erlang OTP平台上&#xff0c;因此安装Rabbit MQ的前提是安装Erlang。通过下面两个连接可以下载安装最新的版本&#xff1a; 下载并安装 Eralng OTP For Windows otp_win64_18.3.exe&#xff08;erlang的环境&#xff09;运行安装 Rabbit MQ Serve…

C++笔记(3)——string.h相关的一些小知识

strlen() 用于得到字符数组中第一个\0前的字符的个数&#xff0c;格式如下&#xff1a; strlen(数组); 例子&#xff1a; #include <stdio.h> #include <string.h>int main(){char str[10];gets(str);int len strlen(str);printf("%d\n", len);return 0…

Java语言与sikuli配合

很早之前写过一篇介绍sikuli的文章。本文简单介绍如何在java中使用sikuli进自动化测试。 图形脚本语言sikuli sikuli IDE可以完成常见的单击、右击、移动到、拖动等鼠标操作&#xff0c;java引用sikuli-script.jar同样可以执行这些常见的鼠标操作&#xff0c;因此即可方便的编写…

angular6 iframe应用

问题一、 iframe如何自适应屏幕高度 解决思路&#xff1a;通过设置iframe外层父元素高度等于window高度&#xff0c;再相对于父元素定位iframe元素&#xff1b;案例如下&#xff1a; 第一步: 模板文件中使用iframe // demo.component.html <div style"position: relati…

jquery下载地址:https://code.jquery.com/jquery/ 影响范围: 版本低于1.7的jQuery过滤用户输入数据所使用的正则表达式存在缺陷,可能导致LOCA

jquery下载地址&#xff1a;https://code.jquery.com/jquery/ 影响范围&#xff1a; 版本低于1.7的jQuery过滤用户输入数据所使用的正则表达式存在缺陷&#xff0c;可能导致LOCATION.HASH跨站漏洞 已测试成功版本&#xff1a; jquery-1.6.min.js&#xff0c;jquery-1.6.1.min…

Myeclipse常用快捷键

2019独角兽企业重金招聘Python工程师标准>>> Ctrl1 快速修复 CtrlD: 删除当前行 CtrlQ 定位到最后编辑的地方 CtrlL 定位在某行 CtrlO 快速显示 OutLine CtrlT 快速显示当前类的继承结构 CtrlW 关闭当前Editer CtrlK 快速定位到下一个 CtrlE 快速显示当前Edi…

数字三角形

问题描述 &#xff08;图&#xff13;.&#xff11;&#xff0d;&#xff11;&#xff09;示出了一个数字三角形。 请编一个程序计算从顶至底的某处的一条路径&#xff0c;使该路径所经过的数字的总和最大。●每一步可沿左斜线向下或右斜线向下走&#xff1b;●1&#xff1c;三…

版本低于1.7的jQuery过滤用户输入数据所使用的正则表达式存在缺陷

jquery下载地址&#xff1a;https://code.jquery.com/jquery/ 影响范围&#xff1a; 版本低于1.7的jQuery过滤用户输入数据所使用的正则表达式存在缺陷&#xff0c;可能导致LOCATION.HASH跨站漏洞 已测试成功版本&#xff1a; jquery-1.6.min.js&#xff0c;jquery-1.6.1.min.…

RabbitMQ学习总结(6)——消息的路由分发机制详解

2019独角兽企业重金招聘Python工程师标准>>> 一、Routing(路由) (using the Java client)在前面的学习中&#xff0c;构建了一个简单的日志记录系统&#xff0c;能够广播所有的日志给多个接收者&#xff0c;在该部分学习中&#xff0c;将添加一个新的特点&#xff0…

Kaggle爆文:一个框架解决几乎所有机器学习问题

上周一个叫 Abhishek Thakur 的数据科学家&#xff0c;在他的 Linkedin 发表了一篇文章 Approaching (Almost) Any Machine Learning Problem&#xff0c;介绍他建立的一个自动的机器学习框架&#xff0c;几乎可以解决任何机器学习问题&#xff0c;项目很快也会发布出来。 这篇…

C# HttpWebRequest GET HTTP HTTPS 请求

这个需求来自于我最近练手的一个项目&#xff0c;在项目中我需要将一些自己发表的和收藏整理的网文集中到一个地方存放&#xff0c;如果全部采用手工操作工作量大而且繁琐&#xff0c;因此周公决定利用C#来实现。在很多地方都需要验证用户身份才可以进行下一步操作&#xff0c;…

Mysql学习总结(4)——MySql基础知识、存储引擎与常用数据类型

2019独角兽企业重金招聘Python工程师标准>>> 1、基础知识 1.1、数据库概述 简单地说&#xff1a;数据库&#xff08;Database或DB&#xff09;是存储、管理数据的容器&#xff1b;严格地说&#xff1a;数据库是“按照某种数据结构对数据进行组织、存储和管理的容器”…

django权限二(多级菜单的设计以及展示)

多级权限菜单设计级标题栏 我们现在只有数据展示,要进入其他url还需要手动的输入路径,非常的麻烦,所以我们要设计 一个导航栏以及侧边多级菜单栏,这个展示是通过stark组件的设计的增删改查页面,而 每一个 页面我们都需要有导航栏和侧边的权限菜单栏,所以把这个公共的部分提起到…

腾讯手游如何提早揭露游戏外挂风险?

目前腾讯SR手游安全测试限期开放免费专家预约&#xff01;点击链接&#xff1a;手游安全测试立即预约&#xff01; 作者&#xff1a;sheldon&#xff0c;腾讯高级安全工程师 商业转载请联系腾讯WeTest获得授权&#xff0c;非商业转载请注明出处。 文中动图无法显示&#xff0c…

下面介绍一个开源的OCR引擎Tesseract2。值得庆幸的是虽然是开源的但是它的识别率较高,并不比其他引擎差劲。网上介绍Tessnet2也是当时时间排名第三的识别引擎,只是后来慢慢不维护了,目前是G

下面介绍一个开源的OCR引擎Tesseract2。值得庆幸的是虽然是开源的但是它的识别率较高&#xff0c;并不比其他引擎差劲。网上介绍Tessnet2也是当时时间排名第三的识别引擎&#xff0c;只是后来慢慢不维护了&#xff0c;目前是Google在维护&#xff0c;大家都知道Google 在搞电子…

分UV教程

第一步 首先&#xff0c;打开一个练习场景“空中预警机1.max”&#xff08;这事小弟平时的练习做的不好献丑了&#xff09;。&#xff08;图01&#xff09; 图01 第二步 这里我们拿机翼来举例子&#xff0c;隐藏除机翼意外的其他模型。&#xff08;图02&#xff09; 图02 第三步…

k8s系列--- dashboard认证及分级授权

http://blog.itpub.net/28916011/viewspace-2215214/ 因版本不一样&#xff0c;略有改动 Dashboard官方地址&#xff1a; https://github.com/kubernetes/dashboard dashbord是作为一个pod来运行&#xff0c;需要serviceaccount账号来登录。 先给dashboad创建一个专用的认证信息…

JAVA项目开发

16年java软件开发经验&#xff0c;全职项目开发&#xff0c;项目可签合同、开普票和专票。 主要承接项目&#xff1a; 1、网站开发项目 自主开发千帆CMS动态发布系统&#xff0c;基于java/springboot2/jpa/easyui开发&#xff0c;简单易用&#xff0c;后台与前端分离&#xff0…

unity3d 任务头上的血条

人物的名称与血条的绘制方法很简单&#xff0c;但是我们需要解决的问题是如何在3D世界中寻找合适的坐标。因为3D世界中的人物是会移动的&#xff0c;它是在3D世界中移动&#xff0c;并不是在2D平面中移动&#xff0c;但是我们需要将3D的人物坐标换算成2D平面中的坐标&#xff0…