博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
设计模式——策略模式
阅读量:5930 次
发布时间:2019-06-19

本文共 2590 字,大约阅读时间需要 8 分钟。

设计模式——策略模式

定义一组算法,将每个算法都封装起来,并且使得他们之间可以相互替换。

先从一个小例子我们慢慢引出策略模式。例如我们经常初学Java的时候肯定想着做一个计算器,简单的加减乘除运算,现在我们只考虑简单的两个数字的加减。有非常多的实现方式,但是我们初学时脑海中第一浮现的应该是下面的代码。

class Calculator{        private final static String ADD = "+";    private final static String SUB = "-";        public int exec(int a, int b,String symbol){        int result = 0;        if (ADD.equals(symbol)){            result = add(a,b);        }else if (SUB.equals(symbol)){            result = sub(a,b);        }        return result;    }        private int add(int a,int b){        return a+b;    }    private int sub(int a,int b){        return a-b;    }}复制代码

此时我们想要优化代码,因为根据上面对于策略模式的定义,我们可以抽象出其实add和sub方法都是一个算法,我们可以将其抽象出来。

interface Calculator{    public int exec(int a,int b);}//加法class AddCalculator implements Calculator{    @Override    public int exec(int a, int b) {        return a+b;    }}//减法class SubCalculator implements Calculator{    @Override    public int exec(int a, int b) {        return a-b;    }}复制代码

根据定义我们将加法和减法两个算法都封装了起来,那么后面的可以使其相互替换是什么意思呢?其实就是根据传入不同的算法执行相应不同的方法,那么根据此来定义一个中间人进行转换。

class Context{        private Calculator calculator = null;        public Context(Calculator calculator){        this.calculator = calculator;    }        public int exec(int a, int b){        return this.calculator.exec(a,b);    }}复制代码

这样在使用的时候,可以对于不同算法的转换。

Context context = null;        context = new Context(new AddCalculator());        System.out.println(context.exec(3,2));        context = new Context(new SubCalculator());        System.out.println(context.exec(3,2));复制代码

那么策略模式的类图我们也可以由此推断出来。

上面有三个角色

  • Strategy:抽象的策略模式。里面定义了每个具体的策略类要实现的定义。
  • ConcreteStrategy:具体的操作。
  • Context:承上启下的作用,传入具体的策略类,执行不同的方法。

策略模式就是这么简单,它就是采用了面向对象的继承和多态的机制。其他的没什么玄机。

其实上面的加减法我们还可以进一步的优化,这里就提出了策略枚举的说法。

enum Calculator {    ADD("+") {        @Override        public int exec(int a, int b) {            return a + b;        }    },     SUB("-") {        @Override        public int exec(int a, int b) {            return a - b;        }    };    String value = "";    private Calculator(String value) {        this.value = value;    }    public String getValue() {        return value;    }    public abstract int exec(int a, int b);}复制代码

我们再进行调用的话就更加简单了。

System.out.println(Calculator.ADD.exec(3,2));System.out.println(Calculator.SUB.exec(3,2));复制代码

为什么上面的代码叫做策略枚举你呢?枚举没问题,因为他是Enum类,那么为什么是策略呢?找找看能不能找到策略的影子在里面。是的我们定义了一个抽象的exec方法。然后在每个枚举成员里面进行了实现。这样是不是和策略模式定义是一样的,对于算法进行了封装有各自实现,并且能够切换。

注意:策略枚举是一个非常优秀和方便的模式,但是它受到枚举类型的限制,每个枚举项都是public、final、static的,扩展性受到了一定的约束,因此在开发中,策略枚举一般担当不经常发生变化的角色。

参考

转载于:https://juejin.im/post/5cbd7b7bf265da03a85abfaf

你可能感兴趣的文章
人们不必担忧数据中心能耗增长过快
查看>>
这款芯片1+1都算不准,但能解决很多大难题
查看>>
特朗普推翻美国ISP隐私政策 允许电信运营商销售用户数据
查看>>
Slack 的野心:企业版的 App Store?
查看>>
雅虎投资人开药方 引入战略投资人进行市值管理
查看>>
CIO如何成为合格管理者
查看>>
高新区以“大数据+大生态”推进“千园之城”建设
查看>>
尼泊尔将在珠峰峰顶提供免费WiFi服务
查看>>
真正的智能家居离我们还有多远
查看>>
Salesforce的Quip收购了一家游戏公司?原来是看上了团队!
查看>>
安防如何从“如何看得清”到“怎样看得明白
查看>>
传微信要开直播业务:主流社交工具全面杀入直播战场
查看>>
Silverlight+WCF 新手实例 象棋 WCF通讯跨域(十五)
查看>>
“智慧城市”提升桂林人生活品质
查看>>
网络安全初创企业Illumio融资1.25亿美元 摩根领投
查看>>
NB-IoT成为利器,物联网云服务平台如何卡位?
查看>>
倾角传感器原理及市场现状浅析
查看>>
数据中心围笼的7个最佳实践
查看>>
布局智能家居:海尔、三星渐行渐远 谁走对了方向?
查看>>
智能家居风越吹越烈 离我们还有多远?
查看>>