设计模式七: 策略(Strategy)
简介
策略属于行为型模式的一种,策略模式允许对象的行为或算法在运行时改变,使用不同的算法达成相同的结果或目的.
实现层面上,定义一个抽象的算法接口, 然后根据具体算法的不同定义不同的类去实现该接口, 调用时不同实现的实例可以相互替换.
策略模式适用于多个类只有行为不同,运行时根据上下文决定使用哪种行为; 根据实际情况不同算法可能不同, 或算法的最优实现在未来确定.
意图
定义一系列算法,封装每个算法,并使它们可以互换。
类图

实现
以绝地求生游戏为例, 要想赢得最后的比赛可能由多种方式(策略), 抽象出 策略接口 WinStrategy, 实现策略 比如激进点的方式AlwaysKillWinStrategy,猥琐点的方式AlwaysHideWinStrategy.
下面我们将实现策略模式(一,二,三步), 并最后讨论更具实际意义的策略与上下文通信的两种方式及其各自利弊(四).
一. 定义策略接口
/**
* 赢得吃鸡的策略接口
*/
public interface WinStrategy {
void win();
}
二. 实现策略,使用两种不同的算法实现
/**
* 采用人挡杀人的模式
*/
public class AlwaysKillWinStrategy implements WinStrategy {
public void win() {
System.out.println("总是很钢,大杀四方");
}
}
/**
* 采用LYB的模式
*/
public class AlwaysHideWinStrategy implements WinStrategy {
public void win() {
System.out.println("总是很苟,活得长久");
}
}
三. 调用
public class StrategyTest {
public static void main(String[] args) {
WinStrategy iWillWin = new AlwaysHideWinStrategy();
iWillWin.win();
iWillWin = new AlwaysKillWinStrategy();
iWillWin.win();
}
}
四. 策略与上下文
上面的例子中我们并没有体现出上下文的概念, 其隐藏的上下文在main()函数中. 除了 策略接口 和 实现策略 外, 还需要一个 策略上下文 角色.一般由上下文持有策略的引用, 并对策略进行实际调用,下面的代码定义该角色
/**
* 上下文
*/
public class Context {
private String weather;
WinStrategy winStrategy;
public Context(String weather, WinStrategy winStrategy) {
this.weather = weather;
this.winStrategy = winStrategy;
}
public void winGame() {
winStrategy.win();
System.out.println("天气情况:"+this.weather+"; 采用的策略:"+winStrategy.getClass());
}
}
//调用
public class StrategyTest {
public static void main(String[] args) {
WinStrategy iWillWin = new AlwaysHideWinStrategy();
Context context = new Context("雾天",iWillWin);
context.winGame();
}
}
策略中有时候也需要某些上下文信息, 例如win()方法需要游戏场景中的天气情况, 这种情况下可以将上下文引用传入策略, 代码修改如下:
/**
* 赢得吃鸡的策略接口,上下文作为参数传入
*/
public interface WinStrategy {
void win(Context context);
}
/**
* 采用LYB的模式, 打印出天气情况
*/
public class AlwaysHideWinStrategy implements WinStrategy {
public void win(Context context) {
System.out.println("总是很苟,活得长久; 天气: "+context.getWeather());
}
}
/**
* 上下文
*/
@Data
public class Context {
private String weather;
WinStrategy winStrategy;
public Context(String weather, WinStrategy winStrategy) {
this.weather = weather;
this.winStrategy = winStrategy;
}
public void winGame() {
winStrategy.win(this);
System.out.println("天气情况:"+this.weather+"; 采用的策略"+winStrategy.getClass());
}
}
上面通过将上下文引用作为参数传递给策略来实现了策略与上下文的通信, 对于不同策略的实现,未必都会用到上下文中的全部信息, 这种情况下传递上下文引用成为了一种"浪费". 因此有的人更愿意将使用的信息作为属性定义在策略实现类中,实现更有针对性的代码
两种方式各有利弊, 传递上下文引用的方式利于扩展, 且可以做到风格统一, 但可能会导致"浪费", 如果扩展太多上下文子类不利于理解. 注入的方式实现在策略类中方式比较简单, 容易实现, 但也会导致策略与其他策略风格不一致.
/**
* 赢得吃鸡的策略接口
*/
public interface WinStrategy {
void win();
}
/**
* 采用人挡杀人的模式
*/
@AllArgsConstructor
public class AlwaysKillWinStrategy implements WinStrategy {
private String weather;
public void win() {
System.out.println("总是很钢,大杀四方"+" 天气:"+weather);
}
}
public class StrategyTest {
public static void main(String[] args) {
WinStrategy iWillWin = new AlwaysKillWinStrategy("雾天");
Context context = new Context("雾天",iWillWin);
context.winGame();
}
}
总结
优点: 1. 易于扩展;2.避免了if...else... 3. 代码重用
缺点: 调用方需要知道使用的哪种算法, 算法需要完全暴露; 可能会导致实现类过多;
设计模式七: 策略(Strategy)的更多相关文章
- C++设计模式实现--策略(Strategy)模式
版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/L_Andy/article/details/30489331 一. 举例说明 曾经做了一个程序,程序 ...
- Java设计模式之 — 策略(Strategy)
转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/8986285 今天你的leader兴致冲冲地找到你,希望你可以帮他一个小忙,他现在急 ...
- Head First 设计模式 - 01. 策略 (Strategy) 模式
当涉及到"维护"时,为了"复用"目的而使用继承,结局并不完美 P4 对父类代码进行修改时,影响层面可能会很大 思考题 利用继承来提供 Duck 的行为,这会导致 ...
- 设计模式 策略-Strategy,装饰-Decorator,观察者-Observer
重温了Head First 3个设计模式.提炼一下思想,笔记如下. 策略-Strategy 当一个类或类族中重复实现某些同类的方法(behavior)时,考虑使用策略模式. 该模式是将Behavior ...
- 乐在其中设计模式(C#) - 策略模式(Strategy Pattern)
原文:乐在其中设计模式(C#) - 策略模式(Strategy Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 策略模式(Strategy Pattern) 作者:webabc ...
- 设计模式C++实现(1)——策略(Strategy)模式
目录 策略模式 应用案例 实现的关键 Talk is cheap,let's See The Code 设计思想 参考 策略模式 策略模式定义了一系列算法和行为(也就是策略),他们可以在运行时相互替换 ...
- Strategy Pattern ava设计模式之策略模式
简介 在策略模式(Strategy Pattern)中,一个类的行为或其算法可以在运行时更改.这种类型的设计模式属于行为型模式.简单理解就是一组算法,可以互换,再简单点策略就是封装算法. 意图 定义一 ...
- JAVA设计模式之策略模式 - Strategy
在策略模式(Strategy Pattern)中,一个类的行为或其算法可以在运行时更改.这种类型的设计模式属于行为型模式. 在策略模式中,我们创建表示各种策略的对象和一个行为随着策略对象改变而改变的 ...
- 设计模式之策略模式和状态模式(strategy pattern & state pattern)
本文来讲解一下两个结构比较相似的行为设计模式:策略模式和状态模式.两者单独的理解和学习都是比较直观简单的,但是实际使用的时候却并不好实践,算是易学难用的设计模式吧.这也是把两者放在一起介绍的原因,经过 ...
随机推荐
- C++ 既有约定
Pascal 拼写法: 函数名MultiplyNumbers(),每个单词的首字母都大写 驼峰拼写法: 变量名 firstNumber,第一个单词的首字母采用小写 匈牙利表示法: iFirstNumb ...
- 011_Python中单线程、多线程和多进程的效率对比实验
Python是运行在解释器中的语言,查找资料知道,python中有一个全局锁(GIL),在使用多进程(Thread)的情况下,不能发挥多核的优势.而使用多进程(Multiprocess),则可以发挥多 ...
- (转)lwip TCP client & FreeRTOS 打开TCP 的 保活机制 LWIP_TCP_KEEPALIVE==1
参考大神教程:http://blog.sina.com.cn/s/blog_62a85b950101aw8x.html 老衲五木 :http://blog.sina.com.cn/s/blog_6 ...
- 服务网关Ocelot 入门Demo系列(01-Ocelot极简单Demo及负载均衡的配置)
[前言] Ocelot是一个用.NET Core实现并且开源的API网关,它功能强大,包括了:路由.请求聚合.服务发现.认证.鉴权.限流熔断.并内置了负载均衡器与Service Fabric.Butt ...
- JShell脚本工具
JShell脚本工具是JDK9的新特性 什么时候会用到 JShell 工具呢,当我们编写的代码非常少的时候,而又不愿意编写类,main方法,也不愿意去编译和运行,这个时候可以使用JShell工具.启动 ...
- day03(变量,常量,输入输出,注释,基本数据类型,运算符)
一,复习 ''' 1.语言的分类 -- 机器语言:直接编写0,1指令,直接能被硬件执行 -- 汇编语言:编写助记符(与指令的对应关系),找到对应的指令直接交给硬件执行 -- 高级语言:编写人能识别的字 ...
- C#中的虚函数及继承关系
转载:http://blog.csdn.net/suncherrydream/article/details/8423991 若一个实例方法声明前带有virtual关键字,那么这个方法就是虚方法. 虚 ...
- centos7之zabbix3.2搭建
环境介绍: centos7.4 zabbix3.2 一.zabbix介绍 zabbix官网:https://www.zabbix.com/ zabbix下载页面:https://www.zabbix. ...
- Starter pom
以下图片是引用书籍内容: 比如你在用boot写一个web项目,在maven中你会导入: <!-- 导入spring boot的web支持 --> <dependency> &l ...
- JS-JAVASCRIPT的eval()方法
Lodop的传统模版是JS语句,如果储存的这种JS模版,可用javascript的eval()方法获取,该方法属于JS的范畴,并不是lodop语句,由于这个方法很多程序员可能不常用,对于这个JS方法不 ...