【重读设计模式】状态模式

2024-03-18 09:38

本文主要是介绍【重读设计模式】状态模式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

状态模式是一个较为常见的模式,因为有限状态机实在是深入人心。那么怎样才是状态模式呢?


定义:当一个对象的内在状态改变时允许改变其行为,这个对象看起来像是改变了其类。

状态模式主要解决的是当控制一个对象状态的条件表达式过于复杂时的情况。把状态的判断逻辑转移到表示不同状态的一系列类中,可以把复杂的判断逻辑简化。
适用场景:
1.一个对象的行为取决于它的状态,并且它必须在运行时刻根据状态改变它的行为。
2.一个操作中含有庞大的多 分支结构,并且这些分支决定于对象的状态。
例子:一般的大系统都有任务系统,比如定时任务。任务有停止、启动、暂停、完成几种状态,而任务会发生状态改变是因行为导致,比如一个任务手动将其启动或者暂停甚至于完成。
类图:

类实现:
//任务类本身
class CTask
{
public: 
CTask():init_state_(NULL),running_state_(NULL),pause_state_(NULL),stop_state_(NULL),curr_state_(NULL)
{
init_state_ = new CInitState(this);
running_state_ = new CRuinningState(this);
pause_state_ = new CPauseState(this);
stop_state_ = new CStopState(this);

//默认的状态是初始状态
curr_state_ = init_state_;
}

//对任务的动作,每个动作都是一个函数,如果按照传统的设计

//用户手动启动一个任务
int start()
{
return curr_state_->start();
}

int stop()
{
return curr_state_->stop();
}

int pause()
{
return curr_state_->pause();
}

int finish()
{
return curr_state_->finish();
}

//设置该任务的状态,注意这里,new_state是一个指针,则替换状态只需要替换指针即可
int set_state(CState* new_state)
{
curr_state_ = new_state;
}

//下面是获取各种状态的实例函数
CState* getInitState()
{
return init_state_;
}
CState* getRunningState()
{
return running_state_;
}
CState* getPauseState()
{
return pause_state_;
}
CState* getStopState()
{
return stop_state_;
}
private:
//任务有下面的几个成员,因为任务有可能是下面的几种状态
CState* init_state_;
CState* running_state_;
CState* pause_state_;
CState* stop_state_;

//任务的当前状态
CState* curr_state_;
}


//状态类,纯抽象类,子类必须实现所有的函数
class CState
{
public:
int start() = 0;
int stop() = 0;
int pause() = 0;
int finish() = 0;
}


//初始状态类
class CInitState : public CState
{
public:
CInitState(CTask* task)
{
task_ = task;
}

//对于初始化的任务来说,下面的动作(也就是函数)的行为是可预计的

//对于初始化任务,是可以有开始操作的
int start()
{
//提示用户可以启动任务,并且将任务状态置为运行中
printf("您启动了任务,该任务当前状态为运行中\n");
task_->set_state(task_->getRunningState());
return 0;
}

int stop()
{
printf("初始状态的任务不支持停止操作\n");
return 0;
}

int pause()
{
printf("初始状态不支持暂停操作\n");
return 0;
}

int finish()
{
printf("您结束了该任务,该任务当前状态为结束\n");
return 0;
}

private:
CTask* task_;
}


//正在运行态
class CRunningState : public CState
{
public:
CRunningState(CTask* task)
{
task_ = task;
}

int start()
{
printf("正在运行中的任务不能再次启动\n");
return 0;
}

int stop()
{
printf("您停止了该任务,该任务当前状态为停止\n");
task_->set_state(task_->getStopState());
return 0;
}

int pause()
{
printf("您暂停了该任务,该任务当前状态为暂停\n");
task_->set_state(task_->getPauseState());
return 0;
}

int finish()
{
printf("正在运行的任务不能被完成\n");
return 0;
}

private:
CTask* task_;
}


//暂停状态
class CPauseState : public CState
{
public:
CPauseState(CTask* task)
{
task_ = task;
}
int start()
{
printf("重启任务成功,该任务当前状态为启动\n");
task_->set_state(task_->getPauseState());
return 0;
}

int stop()
{
printf("停止任务成功,该任务当前状态为停止\n");
task_->set_state(task_->getStopState());
return 0;
}

int pause()
{
printf("处于暂停状态的任务不支持暂停\n");
return 0;
}

int finish()
{
printf("处于暂停状态的任务不支持完成\n");
return 0;
}

private:
CTask* task_;
}


//停止状态
class CStopState : public CState
{
public:
CStopState(CTask* task)
{
task_ = task;
}

int start()
{
printf("处于停止状态的任务不支持启动\n");
return 0;
}

int stop()
{
printf("处于停止状态的任务不支持停止\n");
return 0;
}

int pause()
{
printf("处于停止状态的任务不支持暂停\n");
return 0;
}

int finish()
{
printf("处于停止状态的任务不支持完成\n");
return 0;
}

private:
CTask* task_;
}


class CFinishState : public CState
{
public:
CFinishState(CTask* task)
{
task_ = task;
}

int start()
{
printf("处于完成状态的任务不支持启动\n");
return 0;
}

int stop()
{
printf("处于完成状态的任务不支持停止\n");
return 0;
}

int pause()
{
printf("处于完成状态的任务不支持暂停\n");
return 0;
}

int finish()
{
printf("处于完成状态的任务不支持完成\n");
return 0;
}

private:
CTask* task_;
}


//主函数
void main()
{
//建立一个任务
CTask task;

//启动该任务
task.start();

//暂停该任务
task.pause();

//再启动该任务
task.state();

//停止该任务
task.stop();

return 0;
}

再回头看下状态模式有什么好处,对于普通的面向过程编码方式,应该是这样的:
if(curr_state==STATE_INIT)
{
if(operator == OPER_RUNNING)
{
XXXXX;
}
else if( operator == OPER_PAUSE)
{
xxxx;
}

这样做的后果就是代码逻辑很复杂,不好扩展,比如新增一个状态就要将该函数不断修改,代码复杂度不断上升,而使用状态模式之后,新增状态的复杂度大按大下降,只需要在task类中新增成员和函数,而无需对原有的代码做任何变更。


看了状态模式的定义是不是觉得和另外一个设计模式很相似?对了,状态模式和策略模式很相似,甚至于实现上都差不多。在说明他们的区别之前,我们先将这两个模式都学习完之后再讲述其区别。

这篇关于【重读设计模式】状态模式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/821937

相关文章

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子

SpringBoot如何通过Map实现策略模式

《SpringBoot如何通过Map实现策略模式》策略模式是一种行为设计模式,它允许在运行时选择算法的行为,在Spring框架中,我们可以利用@Resource注解和Map集合来优雅地实现策略模式,这... 目录前言底层机制解析Spring的集合类型自动装配@Resource注解的行为实现原理使用直接使用M

C#原型模式之如何通过克隆对象来优化创建过程

《C#原型模式之如何通过克隆对象来优化创建过程》原型模式是一种创建型设计模式,通过克隆现有对象来创建新对象,避免重复的创建成本和复杂的初始化过程,它适用于对象创建过程复杂、需要大量相似对象或避免重复初... 目录什么是原型模式?原型模式的工作原理C#中如何实现原型模式?1. 定义原型接口2. 实现原型接口3

大数据spark3.5安装部署之local模式详解

《大数据spark3.5安装部署之local模式详解》本文介绍了如何在本地模式下安装和配置Spark,并展示了如何使用SparkShell进行基本的数据处理操作,同时,还介绍了如何通过Spark-su... 目录下载上传解压配置jdk解压配置环境变量启动查看交互操作命令行提交应用spark,一个数据处理框架

Flutter监听当前页面可见与隐藏状态的代码详解

《Flutter监听当前页面可见与隐藏状态的代码详解》文章介绍了如何在Flutter中使用路由观察者来监听应用进入前台或后台状态以及页面的显示和隐藏,并通过代码示例讲解的非常详细,需要的朋友可以参考下... flutter 可以监听 app 进入前台还是后台状态,也可以监听当http://www.cppcn

MySQL 中的服务器配置和状态详解(MySQL Server Configuration and Status)

《MySQL中的服务器配置和状态详解(MySQLServerConfigurationandStatus)》MySQL服务器配置和状态设置包括服务器选项、系统变量和状态变量三个方面,可以通过... 目录mysql 之服务器配置和状态1 MySQL 架构和性能优化1.1 服务器配置和状态1.1.1 服务器选项

linux进程D状态的解决思路分享

《linux进程D状态的解决思路分享》在Linux系统中,进程在内核模式下等待I/O完成时会进入不间断睡眠状态(D状态),这种状态下,进程无法通过普通方式被杀死,本文通过实验模拟了这种状态,并分析了如... 目录1. 问题描述2. 问题分析3. 实验模拟3.1 使用losetup创建一个卷作为pv的磁盘3.

Java实现状态模式的示例代码

《Java实现状态模式的示例代码》状态模式是一种行为型设计模式,允许对象根据其内部状态改变行为,本文主要介绍了Java实现状态模式的示例代码,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来... 目录一、简介1、定义2、状态模式的结构二、Java实现案例1、电灯开关状态案例2、番茄工作法状态案例

通过prometheus监控Tomcat运行状态的操作流程

《通过prometheus监控Tomcat运行状态的操作流程》文章介绍了如何安装和配置Tomcat,并使用Prometheus和TomcatExporter来监控Tomcat的运行状态,文章详细讲解了... 目录Tomcat安装配置以及prometheus监控Tomcat一. 安装并配置tomcat1、安装

Linux之进程状态&&进程优先级详解

《Linux之进程状态&&进程优先级详解》文章介绍了操作系统中进程的状态,包括运行状态、阻塞状态和挂起状态,并详细解释了Linux下进程的具体状态及其管理,此外,文章还讨论了进程的优先级、查看和修改进... 目录一、操作系统的进程状态1.1运行状态1.2阻塞状态1.3挂起二、linux下具体的状态三、进程的