Head First设计模式---1.策略模式

news2025/1/11 12:55:41

4.1策略模式:

策略模式是一种行为设计模式, 它能让你定义一系列算法, 并将每种算法分别放入独立的类中, 以使算法的对象能够相互替换。

策略设计模式

问题

一天,我们需要做一个鸭子游戏,游戏中会出现各种鸭子,一些鸭子会游泳,一些鸭子会嘎嘎叫,一些鸭子会飞,现在我们需要设计这个鸭子应用,我们首先想到的是建立一个鸭子父类,然后让其他鸭子去继承这个父类。

image-20230219165931918

但是随着鸭子的种类越来越多,比如橡皮鸭,他只会叫缺不会飞,我们构造的类已经不满足他的需求了,我们也不能在游戏内让橡皮鸭在天上飞,这个时候怎么办呢?我们可以让fly()方法被覆盖掉,可是如果以后加入了诱饵鸭怎么办呢,他是个木头假鸭,不会飞也不会叫。此时再用继承覆盖会不会以后的类会非常多?

利用接口如何?

如果几个月之后我们还要需求要更新怎么办?所以我们需要一个更清晰的方法,可以让某些(不是全部)鸭子类型可以飞或者可以叫。

image-20230219170444311

那么问题来了,并非所有的子类都可以飞或者都可以叫,所以继承接口不是合适的解决方案。虽然FlyableQuackable可以解决掉一部分问题,但是代码却无法复用,只能是一个噩梦跳进另一个噩梦…

把问题归零

接口可以解决一部分问题,但是却造成了代码无法复用。这意味着:无论你何时修改这个代码,你都需要往下追踪在每一个类中的定义并且去修改它,一不小心就会造成新的错误。

幸运的是,有一个设计模式,可以解决这种问题:找出应用中可能需要变化之处,把他们独立出来,不要和那些不需要变化的代码混在一起

换句话说,如果每次新的需求一来,都需要把某处的代码发生变化,那么你就可以确定,这部分的代码需要被抽取出来,和其他代码有稳定的部分。那么我们是不是可以思考一个问题:”把需要变换的抽取并封装起来,以便以后需要时可以轻易的改变这个部分,而不影响其他变化的部分。“

那么我们需要找到变化和不变化的部分,很简单,我们也知道Duck类的fly()和quack()类会随着鸭子的不同而改变,我们需要建立一组新的类来代表每个行为

image-20230219171418997

那么如何设计鸭子的行为呢?我们需要运动的改变鸭子的飞行行为和呱呱叫的行为,我们在新建对象时需要指定他的各种行为。

有了这些目标,我们看第二个设计原则:针对接口编程,而不是针对实现编程

我们可以用接口代表某个行为:FlyBehavior和QuackBehavior,而行为的每个实现都将实现其中一个接口。

实现鸭子的行为:

image-20230219171754266

整合鸭子的行为:

  1. 首先将FlyBehaviorQuackBehavior抽出来进行封装,因为他们是可变的,并给他们加上对应的属性方法:
public interface QuackBehavior {
    /**
        鸭子呱呱叫的行为,可以不叫 也可以吱吱叫
     */
    void quack();
}
public interface FlyBehavior {
    /**
        实现鸭子飞行 或者什么都不做 不会飞
     */
    void fly();
}
  1. 分别实现FlyBehaviorQuackBehavior对应的行为:
public class Quack implements QuackBehavior{
    @Override
    public void quack() {
        System.out.println("Quack");
    }
}
public class FlyWithWings implements FlyBehavior{
    @Override
    public void fly() {
        System.out.println("I am fly");
    }
}
  1. 接着在Duck类中加入两个实例变量,分别为flyBehaviorquackBehavior,声明为接口类型,每个鸭子都会动态的设置这些变量在运行时引用正确的行为类型

image-20230219172001013

  1. 现在我们来实现performQuack()
//鸭子的种类可以有很多,所以我们将鸭子和他的外观抽象出来:
public abstract class Duck {
    //鸭子的叫声行为
    QuackBehavior quackBehavior;

    //鸭子的飞行行为
    FlyBehavior flyBehavior;

    //鸭子的外观
    public abstract void display();

    public void performQuack(){
        quackBehavior.quack();
    }

    public void performFly(){
        flyBehavior.fly();
    }
}
  1. 我们来关心怎么设置实例变量:绿头鸭类
public class MallardDuck extends Duck{
    public MallardDuck(){
        //绿头鸭类使用呱呱叫
        quackBehavior = new Quack();
        //绿头鸭是可以飞的
        flyBehavior = new FlyWithWings();
    }
    
	@Override
    public void display() {
        System.out.println("I`m a real Mallard duck");
    }
}
  1. 编写测试类
public class Main {
    public static void main(String[] args) {
        Duck mallard = new MallardDuck();
        mallard.performQuack();
        mallard.performFly();
    }
}

运行结果:

Quack
I am fly

但是这样的话就写死,鸭子是可以”动态“扩展的,不然我们设定的那么多动态功能没用用到就太可惜了。我们继续给他加入动态扩展:

  1. 在Duck类中加入两个新的方法:
 	public void setFlyBehavior(FlyBehavior fb){
        flyBehavior = fb;
    }

    public void setQuackBehavior(QuackBehavior qd){
        quackBehavior = qd;
    }
  1. 我们创一个新的鸭子—模型鸭
public class ModelDuck extends Duck{
    public ModelDuck(){
        //模型鸭不会飞
        flyBehavior = new FlyNoWay();
        quackBehavior = new Quack();
    }

    @Override
    public void display() {
        System.out.println("I`m a model duck");
    }
}

public class FlyNoWay implements FlyBehavior{
    @Override
    public void fly() {
        System.out.println("i can`t fly");
    }
}
  1. 我们给模型鸭一个火箭动力,让他飞起来
public class FlyRocketPowered implements FlyBehavior{
    @Override
    public void fly() {
        System.out.println("I`m flying with a rocket");
    }
}
  1. 改变测试类,看看我们的动力鸭:
public class Main {
    public static void main(String[] args) {
        Duck model = new ModelDuck();
        model.performFly();
        model.setFlyBehavior(new FlyRocketPowered());
        model.performFly();
    }
}

运行结果:

i can`t fly
I`m flying with a rocket

所以我们就可以完全自定义。

下面是整个重新设计后的类结构,你所期望的一切都有:鸭子继承Duck,飞行行为实现FlyBehavior接口,呱呱叫行为实现QuackBehavior接口。也请注意,我们描述事情的方式也稍有改变。不再把鸭子的行为说成是“一组行为”,我们开始把行为想成是“一族算法”。想想看,在SimUDuck的设计中,算法代表鸭子能做的事(不同的叫法和飞行法),这样的做法也能很容易地用于用一群类计算不同州的销售税金。

请特别注意类之间的“关系”。拿起笔,把下面图形中的每个箭头标上适当的关系,关系可以是IS-A(是一个)、HAS-A (有一个)或IMPLEMENTS (实现)。

image-20230219173907397

策略模式适合应用场景

当你想使用对象中各种不同的算法变体, 并希望能在运行时切换算法时, 可使用策略模式。

策略模式让你能够将对象关联至可以不同方式执行特定子任务的不同子对象, 从而以间接方式在运行时更改对象行为。

当你有许多仅在执行某些行为时略有不同的相似类时, 可使用策略模式。

策略模式让你能将不同行为抽取到一个独立类层次结构中, 并将原始类组合成同一个, 从而减少重复代码。

如果算法在上下文的逻辑中不是特别重要, 使用该模式能将类的业务逻辑与其算法实现细节隔离开来。

策略模式让你能将各种算法的代码、 内部数据和依赖关系与其他代码隔离开来。 不同客户端可通过一个简单接口执行算法, 并能在运行时进行切换。

当类中使用了复杂条件运算符以在同一算法的不同变体中切换时, 可使用该模式。

策略模式将所有继承自同样接口的算法抽取到独立类中, 因此不再需要条件语句。 原始对象并不实现所有算法的变体, 而是将执行工作委派给其中的一个独立算法对象。

实现方式

  1. 从上下文类中找出修改频率较高的算法 (也可能是用于在运行时选择某个算法变体的复杂条件运算符)。
  2. 声明该算法所有变体的通用策略接口。
  3. 将算法逐一抽取到各自的类中, 它们都必须实现策略接口。
  4. 在上下文类中添加一个成员变量用于保存对于策略对象的引用。 然后提供设置器以修改该成员变量。 上下文仅可通过策略接口同策略对象进行交互, 如有需要还可定义一个接口来让策略访问其数据。
  5. 客户端必须将上下文类与相应策略进行关联, 使上下文可以预期的方式完成其主要工作。

策略模式优缺点

优点:

  • 你可以在运行时切换对象内的算法。

  • 你可以将算法的实现和使用算法的代码隔离开来。

  • 你可以使用组合来代替继承。

  • 开闭原则。 你无需对上下文进行修改就能够引入新的策略。

缺点:

  • 如果你的算法极少发生改变, 那么没有任何理由引入新的类和接口。 使用该模式只会让程序过于复杂。
  • 客户端必须知晓策略间的不同——它需要选择合适的策略。
  • 许多现代编程语言支持函数类型功能, 允许你在一组匿名函数中实现不同版本的算法。 这样, 你使用这些函数的方式就和使用策略对象时完全相同, 无需借助额外的类和接口来保持代码简洁。

与其他模式的关系

  • 桥接模式状态模式策略模式 (在某种程度上包括适配器模式) 模式的接口非常相似。 实际上, 它们都基于组合模式——即将工作委派给其他对象, 不过也各自解决了不同的问题。 模式并不只是以特定方式组织代码的配方, 你还可以使用它们来和其他开发者讨论模式所解决的问题。
  • 命令模式策略看上去很像, 因为两者都能通过某些行为来参数化对象。 但是, 它们的意图有非常大的不同。
    • 你可以使用命令来将任何操作转换为对象。 操作的参数将成为对象的成员变量。 你可以通过转换来延迟操作的执行、 将操作放入队列、 保存历史命令或者向远程服务发送命令等。
    • 另一方面, 策略通常可用于描述完成某件事的不同方式, 让你能够在同一个上下文类中切换算法。
  • 装饰模式可让你更改对象的外表, 策略则让你能够改变其本质。
  • 模板方法模式基于继承机制: 它允许你通过扩展子类中的部分内容来改变部分算法。 策略基于组合机制: 你可以通过对相应行为提供不同的策略来改变对象的部分行为。 模板方法在类层次上运作, 因此它是静态的。 策略在对象层次上运作, 因此允许在运行时切换行为。
  • 状态可被视为策略的扩展。 两者都基于组合机制: 它们都通过将部分工作委派给 “帮手” 对象来改变其在不同情景下的行为。 策略使得这些对象相互之间完全独立, 它们不知道其他对象的存在。 但状态模式没有限制具体状态之间的依赖, 且允许它们自行改变在不同情景下的状态。

文章:First Head设计模式、[设计模式](

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/357345.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

掘金数据时代2022年度隐私计算评选活动火热报名中!

开放隐私计算 开放隐私计算开放隐私计算OpenMPC是国内第一个且影响力最大的隐私计算开放社区。社区秉承开放共享的精神,专注于隐私计算行业的研究与布道。社区致力于隐私计算技术的传播,愿成为中国 “隐私计算最后一公里的服务区”。183篇原创内容公众号…

全网最全虚拟机的封装

1.服务器初始化 系统环境RHEL7.6 2.禁用selinux [rootserver1 ~]# vim /etc/sysconfig/selinux SELINUXdisabled reboot 3.禁用防火墙 [rootserver1 ~]# systemctl disable --now firewalld 4.配置yum源 [rootserver1 ~]# vim /etc/fstab /dev/mapper/rhel…

AC的改进算法——TRPO、PPO

两类AC的改进算法 整理了动手学强化学习的学习内容 1. TRPO 算法(Trust Region Policy Optimization) 1.1. 前沿 策略梯度算法即沿着梯度方向迭代更新策略参数 。但是这种算法有一个明显的缺点:当策略网络沿着策略梯度更新参数&#xff0c…

(考研湖科大教书匠计算机网络)第五章传输层-第五节:TCP拥塞控制

获取pdf:密码7281专栏目录首页:【专栏必读】考研湖科大教书匠计算机网络笔记导航 文章目录一:拥塞控制概述二:拥塞控制四大算法(1)慢开始和拥塞避免A:慢启动(slow start)…

CTFer成长之路之举足轻重的信息搜集

举足轻重的信息搜集CTF 信息搜集 常见的搜集 题目描述: 一共3部分flag docker-compose.yml version: 3.2services:web:image: registry.cn-hangzhou.aliyuncs.com/n1book/web-information-backk:latestports:- 80:80启动方式 docker-compose up -d 题目Flag n1book{in…

设计模式-代理模式

控制和管理访问 玩过扮白脸,扮黑脸的游戏吗?你是一个白脸,提供很好且很友善的服务,但是你不希望每个人都叫你做事,所以找了黑脸控制对你的访问。这就是代理要做的:控制和管理对象。 监视器编码 需求&…

数据挖掘,计算机网络、操作系统刷题笔记49

数据挖掘,计算机网络、操作系统刷题笔记49 2022找工作是学历、能力和运气的超强结合体,遇到寒冬,大厂不招人,可能很多算法学生都得去找开发,测开 测开的话,你就得学数据库,sql,orac…

Spring Cloud Alibaba 微服务简介

微服务简介 1 什么是微服务 2014年,Martin Fowler(马丁福勒 ) 提出了微服务的概念,定义了微服务是由以单一应用程序构成的小服务,自己拥有自己的进程与轻量化处理,服务依业务功能设计,以全自动…

将Nginx 核心知识点扒了个底朝天(四)

为什么 Nginx 不使用多线程? Apache: 创建多个进程或线程,而每个进程或线程都会为其分配 cpu 和内存(线程要比进程小的多,所以 worker 支持比 perfork 高的并发),并发过大会榨干服务器资源。 Nginx: 采用…

程序员35岁中年危机不是坎,是一把程序员自己设计的自旋锁

有时候,我会思考35岁这个程序员的诅咒,确切来说是中国程序员的独有的诅咒。 优秀的程序员思维逻辑严谨,弄清楚需求的本质是每天重复的工作,也是对工作的态度,那弄清楚诅咒的来源,义不容辞。 被诅咒的35岁 …

【爬虫】自动获取showdoc指定项目中的所有文档

▒ 目录 ▒🛫 导读需求1️⃣ 格式分析官方下载文件内容prefix_info.json文件格式2️⃣ 封包分析/api/page/info/api/item/info3️⃣ 编码代码特点问题📖 参考资料🛫 导读 需求 showdoc是一个API文档、技术文档工具网站,经常能搜到…

String intern方法理解

1、原理 参考学习视频: https://www.bilibili.com/video/BV1WK4y1M77t/?spm_id_from333.337.search-card.all.click&vd_source4dc3f886f5ce1d43363b603935f02bd1 String s1 “hello”; String s1 "hello"; 代码原理解释如下图String s1 new Str…

进程章节总结性实验

进程实验课笔记 本节需要有linux基础,懂基本的linux命令操作即可。 Ubuntu镜像下载 https://note.youdao.com/s/VxvU3eVC ubuntu安装 https://www.bilibili.com/video/BV1j44y1S7c2/?spm_id_from333.999.0.0 实验环境ubuntu22版本,那个linux环境都可以…

Linux-VMware常用设置(时间+网络)及网络连接激活失败解决方法-基础篇②

目录一、设置时间二、网络设置1. 激活网卡方法一:直接启动网卡(仅限当此)方法二:修改配置文件(永久)2. 将NAT模式改为桥接模式什么是是NAT模式?如何改为桥接模式?三、虚拟机网络连接…

20230219 质心和重心的区别和性质

质心:(无需重力场的前提)所有质点的位置关于它们的质量的加权平均数。 重心:(需要重力场的前提)重力对系统中每个质点关于重心的力矩之和为零。 质心: xˉ∑i1nmixi∑i1nmi,yˉ∑i1nmiyi∑i1nmi…

Fiddler的报文分析

目录 1.Statistics请求性能数据 2.检测器(Inspectors) 3.自定义响应(AutoResponder) 1.Statistics请求性能数据 报文分析: Request Count: 1 请求数,该session总共发的请求数 Bytes …

vue3.0 生命周期

目录前言:vue3.0生命周期图例1.beforeCreate2.created3.beforeMount/onBeforeMount4.mounted/onMounted5.beforeUpdate/onBeforeUpdate6.updated/onUpdated7.beforeUnmount/onBeforeUnmount8.unmounted/onUnmounted案例:总结前言: 每个Vue组…

智慧城市应急指挥中心数字化及城市驾驶舱建设方案

目 录 第一章 项目概述 1.1 项目背景 1.2 项目范围 第二章 建设内容 2.1 三维可视化平台 2.1.1 多源数据接入 2.1.2 可视化编排 2.1.3 三维可视化编辑 2.1.4 空间数据可视化 2.1.5 集成框架支持 2.2 可视化场景定制开发 2.2.1 城市驾驶总舱 2.2.2 城市安全分舱 2.…

PLT/PDF转CAD:scViewerX 8.1 Crack

scViewerX是一个功能强大的 ActiveX 控件,允许您查看、打印和转换 PLT、Adobe PDF、Autodesk DWF、CGM、Calcomp、HPGL/2、Gerber、TIF、CALS 和其他几种格式。 ScViewerX 可以将您的文件转换为多种不同的输出文件格式,包括 PDF、PDF/A、TIFF、DXF、DWF、…

【人工智能AI】三、NoSQL 实战《NoSQL 企业级基础入门与进阶实战》

帮我写一篇介绍NoSQL的技术文章,文章标题是《NoSQL 实战》,不少于3000字。这篇文章的目录是 3.NoSQL 实战 3.1 MongoDB 入门 3.1.1 MongoDB 基本概念 3.1.2 MongoDB 安装与配置 3.1.3 MongoDB 数据库操作 3.2 Redis 入门 3.2.1 Redis 基本概念 3.2.2 Red…