设计模式:HelloWorld之策略模式
一.概述
策略模式 定义了算法族,分别封装起来,让他们可以互相替换,此模式让算法的变化独立于使用算法的客户。
策略模式的三要素:
抽象策略角色: 策略类,通常由一个接口或者抽象类实现。
具体策略角色:包装了相关的算法和行为。
环境角色:持有一个策略类的引用,最终给客户端调用。
二.案例驱动
提出问题:要求做出一套模拟鸭子的游戏,游戏中会出现各种鸭子,它们一边游泳,一边呱呱叫。
分析:根据OO设计思想,无非就是使各种鸭子实现自己相应的功能即可,如鸭子游泳,鸭子呱呱叫。
解决方案1.0:设计一个接口(Duck),然后根据需要完成不同的实现,如红头鸭,绿头鸭。。。。
public interface Duck {
public abstract void quack();
public abstract void swim();
public abstract void display();
}
public MallardDuck implements Duck{
public abstract void quack(){
System.out.println("呱呱~~");
}
public abstract void swim(){
System.out.println("欢快的游泳~~");
}
public abstract void display(){
System.out.println("绿头鸭~~");
}
}
public RedHeadDuck implements Duck{
public abstract void quack(){
System.out.println("呱呱~~");
}
public abstract void swim(){
System.out.println("欢快的游泳~~");
}
public abstract void display(){
System.out.println("红头鸭~~");
}
}
突然增加一个新的需求,要求有野鸭,除了具有这两个功能外,还会飞(fly),同时还有一种橡皮鸭,叫声时吱吱。。
此时该方案便暴露了一下缺点。。。。
缺点:牵一发而动全身,我们很难知道所有鸭子的行为,当针对有些实现需要添加新的功能时
解决方案1.1
为了更加灵活,面向接口编程。
public interface Flyable {
public abstract void fly();
}
public interface Swimable {
public abstract void swim();
}
public interface Quackable {
public abstract void quack();
}
public RedHeadDuck implements Quackable implements Swimable {...}
public WildDuck implements Quackable implements Swimable implements Flyable {...}
如果有一万种鸭子,这种方式简直不敢想象,重复代码太多!
解决方案2.0
采用策咯模式解决此问题。
step1:把会变化的部分取出来,并封装起来,好让其他代码不受影响。

step2:封装行为的大局观

step3:代码实现
抽象策略角色
// 飞行行为
public interface FlyBehavior {
public abstract void fly();
}
// 叫声行为
public interface QuackBehavior {
public abstract void quack();
}
具体策略对象
// 橡皮鸭叫
public RubberDuckBehavior implements QuackBehavior {
public abstract void quack(){
System.out.println("吱吱~~");
}
}
// 野鸭叫
public WildDuckBehavior implements QuackBehavior {
public abstract void quack(){
System.out.println("嘎嘎~~");
}
}
// 大黄鸭叫
public YellowDuckBehavior implements QuackBehavior {
public abstract void quack(){
System.out.println("呱呱~~");
}
}
// 不会飞
public FlyNoWay implements FlyBehavior {
public abstract void fly(){
System.out.println("不会飞~~");
}
}
// 用翅膀飞
public FlyWithWings implements FlyBehavior {
public abstract void fly(){
System.out.println("用翅膀飞~~");
}
}
// 螺旋桨飞
public FlyLikePlane implements FlyBehavior {
public abstract void fly(){
System.out.println("用螺旋桨~~");
}
}
环境角色
抽象类
public abstract class Duck {
FlyBehavior flyBehavior;
QuackBehavior quackBehavior;
// 动态设定行为
public void setFlyBehavior(FlyBehavior flyBehavior) {
this.flyBehavior = flyBehavior;
}
public void setQuackBehavior(QuackBehavior quackBehavior) {
this.quackBehavior = quackBehavior;
}
public void performFly() {
flyBehavior.fly();
}
public void performQuack() {
quackBehavior.quack();
}
public abstract display() {}
public void swim() {
System.out.println("鸭子天生会游泳!");
}
}
实现
public class WildDuck implements Duck {
FlyBehavior flyBehavior;
QuackBehavior quackBehavior;
public WildDuck() {
// 默认会飞,野鸭叫嘎嘎
this.flyBehavior = new FlyWithWings();
this.quackBehavior = new WildDuckBehavior();
}
// 动态设定行为
public void setFlyBehavior(FlyBehavior flyBehavior) {
this.flyBehavior = flyBehavior;
}
public void setQuackBehavior(QuackBehavior quackBehavior) {
this.quackBehavior = quackBehavior;
}
public void performFly() {
flyBehavior.fly();
}
public void performQuack() {
quackBehavior.quack();
}
public abstract display() {
System.out.println("这是一只野鸭!");
}
public void swim() {
System.out.println("鸭子天生会游泳!");
}
}
需求改变:野鸭的翅膀受伤,不会飞了
public static void main(String[] args) {
Duck duck = new WildDuck();
duck.setFlyBehavior(new FlyNoWay());
duck.performFly();// 输出:不会飞~~
}
需改改变:新增一只火箭鸭,能飞到太空,不会游泳,外形像火箭一样,叫嘎嘎
public RocketDuckFlyBehavior implements FlyBehavior {
public abstract void fly(){
System.out.println("飞到太空~~");
}
}
public class RocketDuck implements Duck {
FlyBehavior flyBehavior;
QuackBehavior quackBehavior;
public WildDuck() {
// 默认会飞,野鸭叫嘎嘎
this.flyBehavior = new RocketDuckFlyBehavior();
this.quackBehavior = new WildDuckQuackBehavior();
}
// 动态设定行为
public void setFlyBehavior(FlyBehavior flyBehavior) {
this.flyBehavior = flyBehavior;
}
public void setQuackBehavior(QuackBehavior quackBehavior) {
this.quackBehavior = quackBehavior;
}
public void performFly() {
flyBehavior.fly();
}
public void performQuack() {
quackBehavior.quack();
}
public abstract display() {
System.out.println("外形像火箭~~");
}
public void swim() {
System.out.println("不会游泳");
}
}
三.策略模式的优缺点
优点:
1.策略模式提供了管理相关的算法族的办法。策略类的等级结构定义了一个算法或行为族。恰当使用继承可以把代码转移到符类,从而避免重复代码 。
2.在策略模式中利用组合和委托来让环境角色拥有执行算法的能力,这也是继承的一种更轻便的替代方案。
3.提供了对开放—封闭原则的完美支持,将算法封装在独立的strategy中,使得它们易于切换,易于理解,易于扩展
4.利用组合、委托和多态等技术和思想,可以有效地避免多重条件选择语句
缺点:
1.客户端必须知道所有的策略类,区分他们之间的区别,并自行决定使用哪一个策略类。
2.针对每一种行为情况需要创建一个策略类,造成很多的策略类。
设计模式:HelloWorld之策略模式的更多相关文章
- 【转】设计模式 ( 十八 ) 策略模式Strategy(对象行为型)
设计模式 ( 十八 ) 策略模式Strategy(对象行为型) 1.概述 在软件开发中也常常遇到类似的情况,实现某一个功能有多种算法或者策略,我们可以根据环境或者条件的不同选择不同的算法或者策略来完成 ...
- 设计模式 ( 十八 ) 策略模式Strategy(对象行为型)
设计模式 ( 十八 ) 策略模式Strategy(对象行为型) 1.概述 在软件开发中也经常遇到类似的情况,实现某一个功能有多种算法或者策略,我们能够依据环境或者条件的不同选择不同的算法或者策略来完毕 ...
- 《Head First 设计模式》[01] 策略模式
<Head First 设计模式>(点击查看详情) 1.写在前面的话 之前在列书单的时候,看网友对于设计模式的推荐里说,设计模式的书类别都大同小异,于是自己就选择了Head First系列 ...
- javascript设计模式--策略模式
javascript策略模式总结 1.什么是策略模式? 策略模式的定义是:定义一系列的算法,把他们独立封装起来,并且可以相互替换. 例如我们需要写一段代码来计算员工的奖金.当绩效为a时,奖金为工资的5 ...
- [head first 设计模式] 第一章 策略模式
[head first 设计模式] 第一章 策略模式 让我们先从一个简单的鸭子模拟器开始讲起. 假设有个简单的鸭子模拟器,游戏中会出现各种鸭子,此系统的原始设计如下,设计了一个鸭子超类,并让各种鸭子继 ...
- Python设计模式: 最佳的"策略"模式实践代码
Python设计模式: 最佳的"策略"模式实践代码 今天抽空看了下流畅的python,发现里面介绍了不少python自带的库的使用实例,用起来非常的优雅. 平时用Python来写爬 ...
- [设计模式] javascript 之 策略模式
策略模式说明 定义: 封装一系列的算法,使得他们之间可以相互替换,本模式使用算法独立于使用它的客户的变化. 说明:策略模式,是一种组织算法的模式,核心不在于算法,而在于组织一系列的算法,并且如何去使用 ...
- 设计模式:Strategy 策略模式 -- 行为型
设计模式 策略模式Strategy(对象行为型) 这是几年前写的文字(转载做的笔记更准确些),发觉还是废话多了点. 其实,核心就是5.结构中的UML图 5.1 和 5.2(新增).现在看这张图就觉得一 ...
- 设计模式浅谈----策略模式(c#及java实现)
一.何为策略模式 策略模式是行为型模式的一种,主要用于需要使用不同的算法来处理不同的数据对象时使用,是一种可以在运行时选择算法的设计模式.也称为政策模式. 主要解决:在有多种算法相似的情况下,使用 i ...
随机推荐
- python/零起点(一、列表)
python/零起点(一.列表) 列表(list)list()可以强行转换数据类型为列表,列表是可迭代对象 列表是有序的,且列表是可变的数据类型 列表中的元素可以是(字符串.整型.元祖.列表.字典.集 ...
- jacascript 立即执行函数(IIFE)与闭包
前言:这是笔者学习之后自己的理解与整理.如果有错误或者疑问的地方,请大家指正,我会持续更新! 一直没搞清楚立即执行函数和闭包之间的关系,总结一下: 闭包有很多种理解:访问不到内部作用域,函数就是这样, ...
- 前端开发必备之chrome插件
Chrome浏览器目前是网络上可用的最好的浏览器之一,并且自2011年11月超越了Firefox浏览器之后,已经成为了互联网上占主导地位的浏览器. 本篇文章将与大家分享一些与前端开发有关的实用的Chr ...
- 妙用 scale 与 transfrom-origin,精准控制动画方向
上次发完 不可思议的纯 CSS 导航栏下划线跟随效果 这篇文章之后,很多朋友找我讨论,感叹 CSS 的奇妙. 然后昨天,群里一位朋友问到了一个和这个效果比较类似的效果,问如何 将下面这个动画的下划线效 ...
- ActiveMQ(七)_伪集群和主从高可用使用(转)
本文转自: https://www.cnblogs.com/gossip/p/5977489.html 一.本文目的 介绍如何在同一台虚拟机上搭建高可用的Activemq服务,集群数量 ...
- 使用springcloud zuul构建接口网关
一 微服务网关背景及简介 不同的微服务一般有不同的网络地址,而外部的客户端可能需要调用多个服务的接口才能完成一个业务需求.比如一个电影购票的收集APP,可能回调用电影分类微服务,用户微服务,支付微服 ...
- ML笔记:Deep Learning
非DL:要找好的特征 DL:无需找好的特征,但新问题:要设计好的网络架构
- SQL注入 手注与联合注入
SQL注入,吧sql命令插入到WEB表单,或输入域名或页面亲求的查询字符串,最终达到欺骗服务器执行恶意的SQL命令: 得到数据库用户名和密码 1:在以,{ .asp?id=32( ...
- [BJOI 2011]元素
Description 相传,在远古时期,位于西方大陆的 Magic Land 上,人们已经掌握了用魔法矿石炼制法杖的技术.那时人们就认识到,一个法杖的法力取决于使用的矿石.一般地,矿石越多则法力越强 ...
- [Codeforces 505C]Mr. Kitayuta, the Treasure Hunter
Description The Shuseki Islands are an archipelago of 30001 small islands in the Yutampo Sea. The is ...