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

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

相关文章

Redis Cluster模式配置

《RedisCluster模式配置》:本文主要介绍RedisCluster模式配置,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录分片 一、分片的本质与核心价值二、分片实现方案对比 ‌三、分片算法详解1. ‌范围分片(顺序分片)‌2. ‌哈希分片3. ‌虚

RabbitMQ工作模式中的RPC通信模式详解

《RabbitMQ工作模式中的RPC通信模式详解》在RabbitMQ中,RPC模式通过消息队列实现远程调用功能,这篇文章给大家介绍RabbitMQ工作模式之RPC通信模式,感兴趣的朋友一起看看吧... 目录RPC通信模式概述工作流程代码案例引入依赖常量类编写客户端代码编写服务端代码RPC通信模式概述在R

IIS 7.0 及更高版本中的 FTP 状态代码

《IIS7.0及更高版本中的FTP状态代码》本文介绍IIS7.0中的FTP状态代码,方便大家在使用iis中发现ftp的问题... 简介尝试使用 FTP 访问运行 Internet Information Services (IIS) 7.0 或更高版本的服务器上的内容时,IIS 将返回指示响应状态的数字代

SQL Server身份验证模式步骤和示例代码

《SQLServer身份验证模式步骤和示例代码》SQLServer是一个广泛使用的关系数据库管理系统,通常使用两种身份验证模式:Windows身份验证和SQLServer身份验证,本文将详细介绍身份... 目录身份验证方式的概念更改身份验证方式的步骤方法一:使用SQL Server Management S

Redis高可用-主从复制、哨兵模式与集群模式详解

《Redis高可用-主从复制、哨兵模式与集群模式详解》:本文主要介绍Redis高可用-主从复制、哨兵模式与集群模式的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录Redis高可用-主从复制、哨兵模式与集群模式概要一、主从复制(Master-Slave Repli

一文带你搞懂Redis Stream的6种消息处理模式

《一文带你搞懂RedisStream的6种消息处理模式》Redis5.0版本引入的Stream数据类型,为Redis生态带来了强大而灵活的消息队列功能,本文将为大家详细介绍RedisStream的6... 目录1. 简单消费模式(Simple Consumption)基本概念核心命令实现示例使用场景优缺点2

使用Python实现IP地址和端口状态检测与监控

《使用Python实现IP地址和端口状态检测与监控》在网络运维和服务器管理中,IP地址和端口的可用性监控是保障业务连续性的基础需求,本文将带你用Python从零打造一个高可用IP监控系统,感兴趣的小伙... 目录概述:为什么需要IP监控系统使用步骤说明1. 环境准备2. 系统部署3. 核心功能配置系统效果展

Nginx location匹配模式与规则详解

《Nginxlocation匹配模式与规则详解》:本文主要介绍Nginxlocation匹配模式与规则,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、环境二、匹配模式1. 精准模式2. 前缀模式(不继续匹配正则)3. 前缀模式(继续匹配正则)4. 正则模式(大

SpringSecurity JWT基于令牌的无状态认证实现

《SpringSecurityJWT基于令牌的无状态认证实现》SpringSecurity中实现基于JWT的无状态认证是一种常见的做法,本文就来介绍一下SpringSecurityJWT基于令牌的无... 目录引言一、JWT基本原理与结构二、Spring Security JWT依赖配置三、JWT令牌生成与

关于WebSocket协议状态码解析

《关于WebSocket协议状态码解析》:本文主要介绍关于WebSocket协议状态码的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录WebSocket协议状态码解析1. 引言2. WebSocket协议状态码概述3. WebSocket协议状态码详解3