策略模式—Java实现
1. 现实需求
本人现在负责开发和维护考核督办系统,其中一个模块叫编写工作计划。是工作计划就要有时间,我们的各种提醒都做了,但是还是有人把x月的工作计划内容写到y月,真心无语了!可能有人会说,让客户直接把错误的工作计划全部修改正确不就行了,但是你想客户会愿意吗?客户永远是想做最少的事带来最多的效果。
好吧,那就是直接修改时间。这时问题来了,如果你修改的时间下,已经有了工作计划怎么办?是覆盖,还是合并,还是……。这时我们需要给客户提供策略供其选择。当然,具体业务还有很多限制,这里不具体描述了。可能我们没有想全所有的情况,或客户有了新的需求,这时我们直接新增策略即可,改很少的代码。基本符合我们面向对象原则中的开闭原则(对扩展开放,对修改关系),实现了高内聚低耦合。
2. 策略模式定义
策略模式,又叫算法簇模式,就是定义了不同的算法族,并且之间可以互相替换,此模式让算法的变化独立于使用算法的客户。
3. 设计原则
设计原则是把一个类中经常改变或者将来可能改变的部分提取出来,作为一个接口然后在类中包 含这 个对象的实例,这样类的实例在运行时就可以随意调用实现了这个接口的类的行为。下面是一个例子。
策略模式属于对象行为型模式,主要针对一组算法,将每一个算法封装到具有共同接口的独立的类中,从而使得它们可以相互替换。策略模式使得算法可以在不影响
到客户端的情况下发生变化。通常,策略模式适用于当一个应用程序需要实现一种特定的服务或者功能,而且该程序有多种实现方式时使用。
4. 策略模式中有三个对象:
(1) 环境对象:该类中实现了对抽象策略中定义的接口或者抽象类的引用。
(2) 抽象策略对象:它可由接口或抽象类来实现。
(3) 具体策略对象:它封装了实现同不功能的不同算法。
利用策略模式构建应用程序,可以根据用户配置等内容,选择不同有算法来实现应用程序的功能。具体的选择有环境对象来完成。采用这种方式可以避免由于使用条件语句而带来的代码混乱,提高应用程序的灵活性与条理性。
5. 为了方便大家理解,还是我们以大多网络上的实例讲解。
刘备要到江东娶老婆了,走之前诸葛亮给赵云(伴郎)三个锦囊妙计,说是按天机拆开能解决棘手问题,嘿,还别说,真解决了大问题,搞到最后是周瑜陪了夫人又折兵,那咱们先看看这个场景是什么样子的。
先说说这个场景中的要素:三个妙计,一个锦囊,一个赵云,妙计是亮哥给的,妙计放在锦囊里,俗称就是锦囊妙计嘛,那赵云就是一个干活的人,从锦囊取出妙计,执行,然后获胜。用java程序怎么表现这些呢?
类图:
项目结构:
具体代码
package chengxuyuanzhilu.com.sp; /** * @author 微信公众号:程序员之路 * 首先定义一个策略接口,这是诸葛亮老人家给赵云的三个锦囊妙计的接口。 */ public interface IStrategy { //每个锦囊妙计都是一个可执行的算法。 public void operate(); } package chengxuyuanzhilu.com.sp.imp; import chengxuyuanzhilu.com.sp.IStrategy; /** * @author 微信公众号:程序员之路 * 找乔国老帮忙,使孙权不能杀刘备 */ public class BackDoor implements IStrategy { @Override public void operate() { System.out.println("找乔国老帮忙,让吴国太给孙权施加压力,使孙权不能杀刘备..."); } } package chengxuyuanzhilu.com.sp.imp; import chengxuyuanzhilu.com.sp.IStrategy; /** * @author 微信公众号:程序员之路 * 孙夫人断后,挡住追兵 */ public class BlackEnemy implements IStrategy { @Override public void operate() { System.out.println("孙夫人断后,挡住追兵..."); } } package chengxuyuanzhilu.com.sp.imp; import chengxuyuanzhilu.com.sp.IStrategy; /** * @author 微信公众号:程序员之路 * 求吴国太开个绿灯 */ public class GivenGreenLight implements IStrategy { @Override public void operate() { System.out.println("求吴国太开个绿灯,放行!"); } } package chengxuyuanzhilu.com.sp; /** * @author 微信公众号:程序员之路 * 放策略的锦囊 */ public class Context { //接口的好处就在于它可以指向实现了它的任意实现类 private IStrategy strategy; //通过锦囊的构造函数,去指定具体要选择哪条策略去处理遇到的危险 public Context(IStrategy strategy) { this.strategy = strategy; } public void operate(){ this.strategy.operate(); } } package chengxuyuanzhilu.com.sp; import chengxuyuanzhilu.com.sp.imp.BackDoor; import chengxuyuanzhilu.com.sp.imp.BlackEnemy; import chengxuyuanzhilu.com.sp.imp.GivenGreenLight; /** * @author 微信公众号:程序员之路 * 赵云就是我们现实生活中的客户 */ public class ZhaoYun { /** * 赵云出场了,他根据诸葛亮给他的交代,依次拆开妙计 */ public static void main(String[] args) { Context context; //刚到吴国的时候拆开第一个 System.out.println("----------刚刚到吴国的时候拆开第一个---------------"); context = new Context(new BackDoor()); context.operate();//拆开执行 System.out.println(" "); //当刘备乐不思蜀时,拆开第二个 System.out.println("----------刘备乐不思蜀,拆第二个了---------------"); context = new Context(new GivenGreenLight()); context.operate();//拆开执行 System.out.println(" "); //孙权的小追兵了,咋办?拆开第三个锦囊 System.out.println("----------孙权的小追兵了,咋办?拆开第三个锦囊---------------"); context = new Context(new BlackEnemy()); context.operate();//拆开执行 System.out.println(" "); } }
执行结构: