【30】c++设计模式——>状态模式

news2024/11/24 11:44:33

状态模式概述

状态模式是一种行为型设计模式,它可以让一个对象在其内部状态发生变化时更改其行为。通过将每个状态封装成一个独立的类,我们可以使状态之间互相独立,并且使得状态的切换变得更加灵活、可扩展。(多个状态之间可以相互转换)
在状态模式中,我们通常会定义一个抽象状态类(Abstract State),以及多个具体状态类(Concrete States)。每个具体状态都会实现抽象状态类中定义的各种操作,并且在需要时执行状态转换。
此外,还有一个环境类(Context),它包含了当前状态,并且在状态发生变化时调用各个状态类的方法来实现状态转换。因为这些状态类都实现了同一个接口,所以环境类不需要知道具体状态的细节,只需要调用相应的方法即可。

如何理解状态类和环境类

电视遥控器
开电视学习
关电视睡觉
静音打游戏

电视遥控器的例子

一个电视遥控器。电视遥控器有三种状态:开机状态、关机状态和静音状态。我们可以通过简单地按遥控器上的按钮来更改状态。为了实现这个功能,首先我们需要定义一个抽象状态类,该类将定义所有可能的操作:

class TVState {
public:
    virtual void onButtonPressed(TVRemote* remote) = 0;
    virtual void offButtonPressed(TVRemote* remote) = 0;
    virtual void muteButtonPressed(TVRemote* remote) = 0;
};

TVState中包含了三种操作:开机、关机和静音。这些操作在不同的状态下可能会有不同的实现方式,因此我们需要在具体状态类中进行实现。

// 具体状态类:开机状态
class OnState : public TVState {
public:
    void onButtonPressed(TVRemote* remote) override {
        std::cout << "The TV is already on." << std::endl;
    }

    void offButtonPressed(TVRemote* remote) override {
        std::cout << "Turning off the TV." << std::endl;
        remote->setState(new OffState());
    }

    void muteButtonPressed(TVRemote* remote) override {
        std::cout << "Muting the TV." << std::endl;
        remote->setState(new MuteState());
    }
};

// 具体状态类:关机状态
class OffState : public TVState {
public:
    void onButtonPressed(TVRemote* remote) override {
        std::cout << "Turning on the TV." << std::endl;
        remote->setState(new OnState());
    }

    void offButtonPressed(TVRemote* remote) override {
        std::cout << "The TV is already off." << std::endl;
    }

    void muteButtonPressed(TVRemote* remote) override {
        std::cout << "Cannot mute the TV when it's turned off." << std::endl;
    }
};

// 具体状态类:静音状态
class MuteState : public TVState {
public:
    void onButtonPressed(TVRemote* remote) override {
        std::cout << "Unmuting the TV." << std::endl;
        remote->setState(new OnState());
    }

    void offButtonPressed(TVRemote* remote) override {
        std::cout << "Turning off the TV." << std::endl;
        remote->setState(new OffState());
    }

    void muteButtonPressed(TVRemote* remote) override {
        std::cout << "The TV is already muted." << std::endl;
    }
};

在这些具体状态类中,我们重写了TVState中定义的所有操作,并且在需要时执行状态转换。例如,在开机状态下按下静音键会将遥控器的状态更改为“静音状态”。
接下来,让我们定义环境类(Context):

class TVRemote {
private:
    TVState* currentState;

public:
    TVRemote() {
        currentState = new OffState();
    }

    void setState(TVState* state) {
        currentState = state;
    }

    void pressOnButton() {
        currentState->onButtonPressed(this);
    }

    void pressOffButton() {
        currentState->offButtonPressed(this);
    }

    void pressMuteButton() {
        currentState->muteButtonPressed(this);
    }
};

在环境类中,我们维护了当前状态,并且在状态发生变化时调用相应的具体状态类方法。我们还定义了三个按键操作:开机、关机和静音。
现在,我们可以使用电视遥控器来测试状态模式的实现了:

int main() {
    TVRemote remote;

    remote.pressOnButton();    // Turning on the TV.
    remote.pressOnButton();    // The TV is already on.
    remote.pressMuteButton();  // Muting the TV.
    remote.pressMuteButton();  // The TV is already muted.
    remote.pressOffButton();   // Turning off the TV.
    remote.pressOffButton();   // The TV is already off.
    remote.pressMuteButton();  // Cannot mute the TV when it's turned off.
    remote.pressOnButton();    // Turning on the TV.
    remote.pressMuteButton();  // Unmuting the TV.

    return 0;
}

通过上面的代码,我们可以看到当我们按下不同的键时,电视遥控器的状态会发生相应的变化。

完整代码

remote.cpp

#include "remote.h"
#include "state.h"

TVRemote::TVRemote(TVState* State )
{
    currentState = State;
}

void TVRemote::setState(TVState* state)
{
    currentState = state;
}

void TVRemote::pressOnButton()
{
    currentState->onButtonPressed(this);
}

void TVRemote::pressOffButton()
{
    currentState->offButtonPressed(this);
}

void TVRemote::pressMuteButton()
{
    currentState->muteButtonPressed(this);
}

remote.h

#pragma once
class TVState; //这里没声明,报了一堆错

class TVRemote
{
private:
    TVState* currentState;

public:
    TVRemote(TVState* State);

    void setState(TVState* state);

    void pressOnButton();

    void pressOffButton();

    void pressMuteButton();
};

state.h

#pragma once
#include"remote.h"


class TVState {
public:
    virtual void onButtonPressed(TVRemote* remote) = 0; // 开机
    virtual void offButtonPressed(TVRemote* remote) = 0; // 关机
    virtual void muteButtonPressed(TVRemote* remote) = 0; // 静音
};

// 具体状态类:关机状态
class OffState : public TVState
{
public:
    void onButtonPressed(TVRemote* remote) override;
    void offButtonPressed(TVRemote* remote) override;
    void muteButtonPressed(TVRemote* remote) override;
};

// 具体状态类:开机状态
class OnState : public TVState
{
public:
    void onButtonPressed(TVRemote* remote) override;
    void offButtonPressed(TVRemote* remote) override;
    void muteButtonPressed(TVRemote* remote) override;
};

// 具体状态类:静音状态
class MuteState : public TVState {
public:
    void onButtonPressed(TVRemote* remote) override;
    void offButtonPressed(TVRemote* remote) override;
    void muteButtonPressed(TVRemote* remote) override;
};

state.cpp

#include<iostream>
#include "state.h"
#include "remote.h"


// 具体状态类:关机状态
void OffState::onButtonPressed(TVRemote* remote) 
{
    std::cout << "Turning on the TV." << std::endl;
    remote->setState(new OnState());
}

void OffState::offButtonPressed(TVRemote* remote) 
{
    std::cout << "The TV is already off." << std::endl;
}

void OffState::muteButtonPressed(TVRemote* remote)
{
    std::cout << "Cannot mute the TV when it's turned off." << std::endl;
}

// 具体状态类:开机状态
void OnState::onButtonPressed(TVRemote* remote) 
{
    std::cout << "The TV is already on." << std::endl;
}

void OnState::offButtonPressed(TVRemote* remote) 
{
    std::cout << "Turning off the TV." << std::endl;
    remote->setState(new OffState());
}

void OnState::muteButtonPressed(TVRemote* remote)
{
    std::cout << "Muting the TV." << std::endl;
    remote->setState(new MuteState());
}


// 具体状态类:静音状态
void MuteState::onButtonPressed(TVRemote* remote)
{
    std::cout << "Unmuting the TV." << std::endl;
    remote->setState(new OnState());
}

void MuteState::offButtonPressed(TVRemote* remote) 
{
    std::cout << "Turning off the TV." << std::endl;
    remote->setState(new OffState());
}

void MuteState::muteButtonPressed(TVRemote* remote)
{
    std::cout << "The TV is already muted." << std::endl;
}

main.cpp

#include <iostream>
using namespace std;
#include "state.h"
#include "remote.h"

int main() {
    TVState* off = new MuteState;
    TVRemote remote(off);

    remote.pressOnButton();    // Turning on the TV.
    remote.pressOnButton();    // The TV is already on.
    remote.pressMuteButton();  // Muting the TV.
    remote.pressMuteButton();  // The TV is already muted.
    remote.pressOffButton();   // Turning off the TV.
    remote.pressOffButton();   // The TV is already off.
    remote.pressMuteButton();  // Cannot mute the TV when it's turned off.
    remote.pressOnButton();    // Turning on the TV.
    remote.pressMuteButton();  // Unmuting the TV.

    return 0;
}

在这里插入图片描述

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

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

相关文章

CSS边框线段样式

让我为大家介绍一下边框样式吧&#xff01;如果大家想更进一步了解边框的使用&#xff0c;可以阅读这一篇文章&#xff1a;CSS边框border 属性描述none没有边框,即忽略所有边框的宽度(默认值)solid边框为单实线dashed边框为虚线dotted边框为点线double边框为双实线 代码演示&…

XPS就是分一下峰没你想的那么简单!-科学指南针

还记得前一段时间的一篇刷屏的经典文章吗! 林雪平大学(Linkping University)的Grzegorz Greczynski和Lars Hultman二人发表观点性文章&#xff0c;对诺奖得主K. Siegbahn推荐的XPS校准方法可能存在的问题进行了阐述与批评&#xff0c;并提出建议。文章原标题为“Compromising S…

XPS测试能谱数据如何处理-科学指南针

本文将分三部分介绍如何用ORIGIN软件处理XPS测试能谱数据&#xff1a; 1、多元素谱图数据处理 2、剖面分析数据处理 3、复杂谱图的解叠 一、多元素谱图的处理&#xff1a; 1、将ASC码文件用NOTEPAD打开&#xff1a; 2、复制Y轴数值。打开ORIGIN&#xff0c;将Y轴数据粘贴到B&am…

栈和队列(8.4)

1. 栈 1.1 栈的概念及结构 栈&#xff1a;一种特殊的线性表&#xff0c;其只允许在固定的一端进行插入和删除元素操作。 进行数据插入和删除操作的一端 称为栈顶&#xff0c;另一端称为栈底。 栈中的数据元素遵守后进先出 LIFO &#xff08; Last In First Out &#xff09;…

移远通信联合产业中坚力量共同发起倡议,推动5G RedCap技术演进和应用创新发展

10月20-21日&#xff0c;2023年中国5G发展大会在上海召开。会议第二日&#xff0c;“5G轻量化&#xff08;RedCap&#xff09;技术演进和应用创新发展”论坛同步举办。 作为5G及RedCap技术发展的先行力量&#xff0c;移远通信受邀出席论坛。同时&#xff0c;公司副总经理刘明辉…

Go 类型全解:常量与变量大全!

一、类型确定值 类型确定值是与特定数据类型明确绑定的。类型确定值在 Go 中占有很大一部分领域&#xff0c;包括但不限于常量、变量、函数返回值、结构体字段等。下面是对类型确定值的的示例&#xff1a; 类型确定值在变量声明中 当你在变量声明中明确指定了类型&#xff0…

PLC MODBUS-ASCII协议通信( LRC校验码SCL计算FC)

Modbus-RTU协议在RS485总线上有RTU和ASCII两种传输格式,其中RTU协议应用比较多,ASCII协议很少使用。有些仪表可能会使用到ASCII协议,这篇博客我们简单介绍下MODBUS-ASCII协议,ASCII协议使用的是文本传输,整个数据包是可打印字符。 帧头一般是冒号: 帧尾是换行符\r\n。 有…

【每日一题】老人的数目

文章目录 Tag题目来源题目解读解题思路方法一&#xff1a;遍历 其他语言python3 写在最后 Tag 【遍历】【数组】【2023-10-23】 题目来源 2678. 老人的数目 题目解读 找出乘客中年龄严格大于 60 的人数。 解题思路 方法一&#xff1a;遍历 本题比较简单&#xff0c;直接遍…

智慧燃气巡检管理系统

我们知道燃气设施的巡检、巡查是运维工作中一项重要的基础工作&#xff0c;而巡检人员主要靠手动记录&#xff0c;回到公司后还得再进行录入归档、导入照片&#xff0c;然后打印装订等&#xff0c;涉及工作量也是不小的&#xff1b;还有人员更替&#xff0c;易造成人员对燃气设…

【设计模式】概括

设计模式 什么是设计模式 设计模式&#xff0c;是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性、程序的重用性。 设计模式分类 创建型模式&#xff0c;共五种&#xff1a;工厂…

Java方法区

方法区 Java方法区&#xff08;Method Area&#xff09;&#xff0c;在Java虚拟机&#xff08;JVM&#xff09;内存结构中是一个非常重要的组成部分。方法区是用来存储类信息、常量、静态变量以及即时编译器编译后的代码等数据的内存区域。 内部结构 类元数据&#xff08;Cl…

Python数据挖掘 | 升级版自动查核酸

&#x1f4d5;作者简介&#xff1a;热爱跑步的恒川&#xff0c;致力于C/C、Java、Python等多编程语言&#xff0c;热爱跑步&#xff0c;喜爱音乐的一位博主。 &#x1f4d7;本文收录于恒川的日常汇报系列&#xff0c;大家有兴趣的可以看一看 &#x1f4d8;相关专栏C语言初阶、C…

常见的状态转移矩阵和对应的运动模型

状态转移矩阵的形式取决于我们所建模的系统的动态特性。对于不同的运动模型&#xff0c;状态转移矩阵将会有所不同。以下是一些常见的状态转移矩阵和对应的运动模型&#xff1a; 恒定速度模型&#xff1a; 这是你给出的模型&#xff0c;其中物体假设以恒定速度移动。 恒定加速…

图书推荐管理系统Python+Django网页界面+协同过滤推荐算法

一、介绍 图书管理与推荐系统。使用Python作为主要开发语言。前端采用HTML、CSS、BootStrap等技术搭建界面结构&#xff0c;后端采用Django作为逻辑处理&#xff0c;通过Ajax等技术实现数据交互通信。在图书推荐方面使用经典的协同过滤算法作为推荐算法模块。主要功能有&#…

【python海洋专题二十八】南海四季海流流速图

【python海洋专题二十八】南海四季海流流速图 往期推荐 图片 【python海洋专题一】查看数据nc文件的属性并输出属性到txt文件 【python海洋专题二】读取水深nc文件并水深地形图 【python海洋专题三】图像修饰之画布和坐标轴 【Python海洋专题四】之水深地图图像修饰…

云HIS系统,Cloud HIS system,云HIS医院信息管理系统源码

通过云HIS平台,可以减少医院投资,无需自建机房和系统,快速实现信息化服务。系统升级及日常维护服务有云平台提供,无需配备专业IT维护人员进行系统维护。 一、his系统和云his系统的区别 His系统和云his系统是两种不同的计算平台&#xff0c;它们在技术架构上存在很大的差异。下…

【python海洋专题二十七】南海四季海流图

【python海洋专题二十七】南海四季海流图 往期推荐 **[[ 【python海洋专题一】查看数据nc文件的属性并输出属性到txt文件] 【python海洋专题二】读取水深nc文件并水深地形图 【python海洋专题三】图像修饰之画布和坐标轴 【Python海洋专题四】之水深地图图像修饰 【Pyth…

电动汽车交流充电桩系统的设计方案

摘要&#xff1a;作为新能源汽车的基础动力装置&#xff0c;交流充电桩也是可以促使新能源汽车正常行驶的关键内容。与其他汽车不同的是&#xff0c;新能源汽车并不需要汽油维持其运行&#xff0c;只需要充电就可以保证汽车行驶的需求&#xff0c;可以降低汽油排放对环境的污染…

SpringBoot项目把Mysql从5.7升级到8.0

首先你需要把之前的库导入到mysql库导入到8.0的新库中。&#xff08;导入的时候会报错我是通过navcat备份恢复的&#xff09; 1、项目中需要修改pom文件的依赖 mysql 和 jdbc <dependency><groupId>mysql</groupId><artifactId>mysql-connector-java&…

nginx配置负载均衡--实战项目(适用于轮询、加权轮询、ip_hash)

&#x1f468;‍&#x1f393;博主简介 &#x1f3c5;云计算领域优质创作者   &#x1f3c5;华为云开发者社区专家博主   &#x1f3c5;阿里云开发者社区专家博主 &#x1f48a;交流社区&#xff1a;运维交流社区 欢迎大家的加入&#xff01; &#x1f40b; 希望大家多多支…