Java设计模式之策略模式
2016-03-27 08:52
381 查看
背景
在软件开发中常常遇到这种情况,实现某一个功能有多种算法或者策略,我们可以根据环境或者条件的不同选择不同的算法或者策略来完成该功能。如查找、排序等,一种常用的方法是硬编码(Hard Coding)在一个类中,如需要提供多种查找算法,可以将这些算法写到一个类中,在该类中提供多个方法,每一个方法对应一个具体的查找算法;当然也可以将这些查找算法封装在一个统一的方法中,通过if…else…或者case等条件判断语句来进行选择。
这两种实现方法我们都可以称之为硬编码,如果需要增加一种新的查找算法,需要修改封装算法类的源代码;更换查找算法,也需要修改客户端调用代码。在这个算法类中封装了大量查找算法,该类代码将较复杂,维护较为困难。如果我们将这些策略包含在客户端,这种做法更不可取,将导致客户端程序庞大而且难以维护,如果存在大量可供选择的算法时问题将变得更加严重。
问题
如何让算法和对象分开来,使得算法可以独立于使用它的客户而变化?
方案
把一个类中经常改变或者将来可能改变的部分提取出来,作为一个接口,然后在类中包含这个对象的实例,这样类的实例在运行时就可以随意调用实现了这个接口的类的行为。
比如定义一系列的算法,把每一个算法封装起来, 并且使它们可相互替换,使得算法可独立于使用它的客户而变化。这就是策略模式。
适用情况
许多相关的类仅仅是行为有异。 “策略”提供了一种用多个行为中的一个行为来配置一个类的方法。即一个系统需要动态地在几种算法中选择一种。
当一个应用程序需要实现一种特定的服务或者功能,而且该程序有多种实现方式时使用。
一个类定义了多种行为 , 并且这些行为在这个类的操作中以多个条件语句的形式出现。将相关的条件分支移入它们各自的Strategy类中以代替这些条件语句。
优点
1、可以动态的改变对象的行为
缺点
1、客户端必须知道所有的策略类,并自行决定使用哪一个策略类
2、策略模式将造成产生很多策略类
组成
环境类(Context):用一个ConcreteStrategy对象来配置。维护一个对Strategy对象的引用。可定义一个接口来让Strategy访问它的数据。
抽象策略类(Strategy):定义所有支持的算法的公共接口。 Context使用这个接口来调用某ConcreteStrategy定义的算法。
具体策略类(ConcreteStrategy):以Strategy接口实现某具体算法。
应用
场景如下,刘备要到江东娶老婆了,走之前诸葛亮给赵云三个锦囊妙计,说是按天机拆开能解决棘手问题。场景中出现三个要素:三个妙计(具体策略类)、一个锦囊(环境类)、赵云(调用者)。
抽象策略类(Strategy)
三个实现类(ConcreteStrategy):
妙计一:初到吴国
妙计二:求吴国太开绿灯放行
妙计三:孙夫人断后,挡住追兵
环境类(Context)
下面就是使用的情况了
以上就是策略模式,多种不同解决方案动态切换,起到改变对象行为的效果。
原作者:jason0539
博客:http://blog.csdn.net/jason0539(转载请说明出处)
在软件开发中常常遇到这种情况,实现某一个功能有多种算法或者策略,我们可以根据环境或者条件的不同选择不同的算法或者策略来完成该功能。如查找、排序等,一种常用的方法是硬编码(Hard Coding)在一个类中,如需要提供多种查找算法,可以将这些算法写到一个类中,在该类中提供多个方法,每一个方法对应一个具体的查找算法;当然也可以将这些查找算法封装在一个统一的方法中,通过if…else…或者case等条件判断语句来进行选择。
这两种实现方法我们都可以称之为硬编码,如果需要增加一种新的查找算法,需要修改封装算法类的源代码;更换查找算法,也需要修改客户端调用代码。在这个算法类中封装了大量查找算法,该类代码将较复杂,维护较为困难。如果我们将这些策略包含在客户端,这种做法更不可取,将导致客户端程序庞大而且难以维护,如果存在大量可供选择的算法时问题将变得更加严重。
问题
如何让算法和对象分开来,使得算法可以独立于使用它的客户而变化?
方案
把一个类中经常改变或者将来可能改变的部分提取出来,作为一个接口,然后在类中包含这个对象的实例,这样类的实例在运行时就可以随意调用实现了这个接口的类的行为。
比如定义一系列的算法,把每一个算法封装起来, 并且使它们可相互替换,使得算法可独立于使用它的客户而变化。这就是策略模式。
适用情况
许多相关的类仅仅是行为有异。 “策略”提供了一种用多个行为中的一个行为来配置一个类的方法。即一个系统需要动态地在几种算法中选择一种。
当一个应用程序需要实现一种特定的服务或者功能,而且该程序有多种实现方式时使用。
一个类定义了多种行为 , 并且这些行为在这个类的操作中以多个条件语句的形式出现。将相关的条件分支移入它们各自的Strategy类中以代替这些条件语句。
优点
1、可以动态的改变对象的行为
缺点
1、客户端必须知道所有的策略类,并自行决定使用哪一个策略类
2、策略模式将造成产生很多策略类
组成
环境类(Context):用一个ConcreteStrategy对象来配置。维护一个对Strategy对象的引用。可定义一个接口来让Strategy访问它的数据。
抽象策略类(Strategy):定义所有支持的算法的公共接口。 Context使用这个接口来调用某ConcreteStrategy定义的算法。
具体策略类(ConcreteStrategy):以Strategy接口实现某具体算法。
应用
场景如下,刘备要到江东娶老婆了,走之前诸葛亮给赵云三个锦囊妙计,说是按天机拆开能解决棘手问题。场景中出现三个要素:三个妙计(具体策略类)、一个锦囊(环境类)、赵云(调用者)。
抽象策略类(Strategy)
package com.cn.celiu; //抽象策略类(Strategy) public interface Strategy { public void operate1(); public void operate2(); }
三个实现类(ConcreteStrategy):
妙计一:初到吴国
package com.cn.celiu; public class BackDoor implements Strategy { @Override public void operate1() { // TODO Auto-generated method stub System.out.println("找乔国老帮忙,让吴国太给孙权施加压力,使孙权不能杀刘备1"); } @Override public void operate2() { // TODO Auto-generated method stub System.out.println("找乔国老帮忙,让吴国太给孙权施加压力,使孙权不能杀刘备2"); } }
妙计二:求吴国太开绿灯放行
package com.cn.celiu; public class GivenGreenLight implements Strategy { @Override public void operate1() { // TODO Auto-generated method stub System.out.println("求吴国太开个绿灯,放行1"); } @Override public void operate2() { // TODO Auto-generated method stub System.out.println("求吴国太开个绿灯,放行2"); } }
妙计三:孙夫人断后,挡住追兵
package com.cn.celiu; public class BlackEnemy implements Strategy { @Override public void operate1() { // TODO Auto-generated method stub System.out.println("孙夫人断后,挡住追兵1"); } @Override public void operate2() { // TODO Auto-generated method stub System.out.println("孙夫人断后,挡住追兵2"); } }
环境类(Context)
package com.cn.celiu; public class Context { private Strategy strategy; public Context(Strategy strategy) { this.strategy = strategy; } public Strategy getStrategy() { return strategy; } public void setStrategy(Strategy strategy) { this.strategy = strategy; } public void operate1() { this.strategy.operate1(); } public void operate2() { this.strategy.operate2(); } }
下面就是使用的情况了
package com.cn; import com.cn.celiu.BackDoor; import com.cn.celiu.BlackEnemy; import com.cn.celiu.Context; import com.cn.celiu.GivenGreenLight; public class main { public static void main(String[] args) { // TODO Auto-generated method stub // 策略模式 Context context; System.out.println("----------刚到吴国使用第一个锦囊---------------"); context = new Context(new BackDoor()); context.operate1(); context.operate2(); System.out.println("\n"); System.out.println("----------刘备乐不思蜀使用第二个锦囊---------------"); context.setStrategy(new GivenGreenLight()); context.operate1(); context.operate2(); System.out.println("\n"); System.out.println("----------孙权的追兵来了,使用第三个锦囊---------------"); context.setStrategy(new BlackEnemy()); context.operate1(); context.operate2(); System.out.println("\n"); } }
以上就是策略模式,多种不同解决方案动态切换,起到改变对象行为的效果。
原作者:jason0539
博客:http://blog.csdn.net/jason0539(转载请说明出处)
相关文章推荐
- 《Java虚拟机原理图解》1.4 class文件里的字段表集合--field字段在class文件里是如何组织的
- java学习之---java虚拟机浅谈
- myEclipse 创建服务器
- Java反射机制
- 分布式 Springmvc myBatis shiro restful dubbo zookeepe
- java cp classpath
- Java基本类型
- 20145308刘昊阳 《Java程序设计》第4周学习总结
- Path must include project and resource name: /struts-default.xml
- myeclipse/eclipse修改JSP生成模板pageEncoding为UTF-8
- NetBeans建立跳过测试构建的快捷方式
- Spring 事务管理
- Java并发:可重入与线程安全
- Java并发:如何编写线程安全的代码
- Java重写equals方法和hashCode方法
- J2SE中的继承和封装
- Java获取当前进程ID以及所有Java进程的进程ID
- 三张图彻底了解Java中字符串的不变性
- Java语言基础
- 如何解决eclipse中ADT22升级到ADT23产生冲突的问题