C# 设计模式系列教程-状态模式
作者:Wang Juqiang 发布时间:2022-11-07 13:31:55
标签:C#,设计模式,状态模式
1. 概述
当一个对象的内在状态改变时允许改变其行为,这个对象看起来像是改变了其类。
2. 解决的问题
主要解决的是当控制一个对象状态转换的条件表达式过于复杂时的情况。把状态的判断逻辑转移到表示不同的一系列类当中,可以把复杂的逻辑判断简单化。
3. 模式中的角色
3.1 上下文环境(Context):它定义了客户程序需要的接口并维护一个具体状态角色的实例,将与状态相关的操作委托给当前的Concrete State对象来处理。
3.2 抽象状态(State):定义一个接口以封装使用上下文环境的的一个特定状态相关的行为。
3.3 具体状态(Concrete State):实现抽象状态定义的接口。
4. 模式解读
4.1 状态模式的类图
4.2 状态模式的代码实现
/// <summary>
/// Context类,维护一个ConcreteState子类的实例,这个实例定义当前的状态。
/// </summary>
public class Context
{
private State state;
/// <summary>
/// 定义Context的初始状态
/// </summary>
/// <param name="state"></param>
public Context(State state)
{
this.state = state;
}
/// <summary>
/// 可读写的状态属性,用于读取和设置新状态
/// </summary>
public State State
{
get { return state; }
set { state = value; }
}
/// <summary>
/// 对请求做处理,并设置下一个状态
/// </summary>
public void Request()
{
state.Handle(this);
}
}
/// <summary>
/// 抽象状态类,定义一个接口以封装与Context的一个特定状态相关的行为
/// </summary>
public abstract class State
{
public abstract void Handle(Context context);
}
/// <summary>
/// 具体状态类,每一个子类实现一个与Context的一个状态相关的行为
/// </summary>
public class ConcreteStateA : State
{
/// <summary>
/// 设置ConcreteStateA的下一个状态是ConcreteStateB
/// </summary>
/// <param name="context"></param>
public override void Handle(Context context)
{
Console.WriteLine("当前状态是 A.");
context.State = new ConcreteStateB();
}
}
public class ConcreteStateB : State
{
/// <summary>
/// 设置ConcreteStateB的下一个状态是ConcreteSateA
/// </summary>
/// <param name="context"></param>
public override void Handle(Context context)
{
Console.WriteLine("当前状态是 B.");
context.State = new ConcreteStateA();
}
}
4.3 客户端调用
class Program
{
static void Main(string[] args)
{
// 设置Context的初始状态为ConcreteStateA
Context context = new Context(new ConcreteStateA());
// 不断地进行请求,同时更改状态
context.Request();
context.Request();
context.Request();
context.Request();
Console.Read();
}
}
运行结果
5. 模式总结
5.1 优点
5.1.1 状态模式将与特定状态相关的行为局部化,并且将不同状态的行为分割开来。
5.1.2 所有状态相关的代码都存在于某个ConcereteState中,所以通过定义新的子类很容易地增加新的状态和转换。
5.1.3 状态模式通过把各种状态转移逻辑分不到State的子类之间,来减少相互间的依赖。
5.2 缺点
5.2.1 导致较多的ConcreteState子类
5.3 适用场景
5.3.1 当一个对象的行为取决于它的状态,并且它必须在运行时刻根据状态改变它的行为时,就可以考虑使用状态模式来。
5.3.2 一个操作中含有庞大的分支结构,并且这些分支决定于对象的状态。
6. 应用举例:电灯有两个状态,开(亮)与关(不亮),下面就用状态模式来实现对电灯的控制。
6.1 类图
6.2 实现代码
/// <summary>
/// 电灯类,对应模式中的Context类
/// </summary>
public class Light
{
private LightState state;
public Light(LightState state)
{
this.state = state;
}
/// <summary>
/// 按下电灯开关
/// </summary>
public void PressSwich()
{
state.PressSwich(this);
}
public LightState State
{
get { return state; }
set { state = value; }
}
}
/// <summary>
/// 抽象的电灯状态类,相当于State类
/// </summary>
public abstract class LightState
{
public abstract void PressSwich(Light light);
}
/// <summary>
/// 具体状态类, 开
/// </summary>
public class On : LightState
{
/// <summary>
/// 在开状态下,按下开关则切换到关的状态。
/// </summary>
/// <param name="light"></param>
public override void PressSwich(Light light)
{
Console.WriteLine("Turn off the light.");
light.State = new Off();
}
}
/// <summary>
/// 具体状态类,关
/// </summary>
public class Off: LightState
{
/// <summary>
/// 在关状态下,按下开关则打开电灯。
/// </summary>
/// <param name="light"></param>
public override void PressSwich(Light light)
{
Console.WriteLine("Turn on the light.");
light.State = new On();
}
}
6.3 客户端代码
class Program
{
static void Main(string[] args)
{
// 初始化电灯,原始状态为关
Light light = new Light(new Off());
// 第一次按下开关,打开电灯
light.PressSwich();
// 第二次按下开关,关闭电灯
light.PressSwich();
Console.Read();
}
}
执行结果


猜你喜欢
- 前言在日常开发中,除了修改请求参数、设置响应header,响应body外,还有一种需求就是url重新,或者是修改url,这里简述一下怎么在z
- 废话不多说,咱们第一篇文章就是模仿“知乎”的回答详情页的动画效果,先上个原版的效果图,咱们就是要做出这个效果 &nbs
- XY个人记SparkSQL是spark的一个模块,主入口是SparkSession,将SQL查询与Spark程序无缝混合。DataFrame
- 工欲善其事,必先利其器,对于想要深入学习Android源码,必须先掌握Android编译命令.一、引言关于Android Build系统,这
- Spring容器可以自动装配相互协作bean之间的关系,这有助于减少对XML配置,而无需编写一个大的基于Spring应用程序的较多的<
- 导读导读 | 12月总体来说互联网的技术圈是非常热闹的,chatGPT爆火,SpringBoot3.0发布等重磅陆消息续进入大家的视线,而本
- package com.cjonline.foundation.evisa;import java.io.BufferedReader;im
- 本文实例讲述了C#实现IP摄像头的方法。分享给大家供大家参考。具体实现方法如下:#region IP摄像头代码/// <summary
- 在应用C#进行Winform窗体程序编写的时候,经常需要编写工具栏。下面小编给大家分享一下C#如何应用ToolSctrip控件编写工具栏。1
- 这篇文章主要介绍了Spring boot2X负载均衡和反向代理实现过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参
- 本文实例为大家分享了Android实现折线图小工具的具体代码,供大家参考,具体内容如下1.LineChart类public class Li
- 一、项目概述之前有不少粉丝私信我说,能不能用Android原生的语言开发一款在手机上运行的游戏呢?说实话,使用java语言直接开发游戏这个需
- 目录PdfSharp下载涉及知识点文档示例图核心代码PdfSharp一款开源的用于创建,操作PDF文档的.Net类库,本文以一个简单的小例子
- 一、饿汉式(静态常量)public class Face { private stat
- File类简介package com.file;import java.io.File;import java.io.IOException
- 本文实例为大家分享了Java从服务端下载Excel模板文件的具体实现代码,供大家参考,具体内容如下方法一 (2021年01月更新)生成exc
- 这篇文章主要介绍了SpringBoot多模块项目框架搭建过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值
- 一说到写日志,大家可能推荐一堆的开源日志框架,如:Log4Net、NLog,这些日志框架确实也不错,比较强大也比较灵活,但也正因为又强大又灵
- 事发地原默认的Feign是使用URLConnector进行通信的,当换为okhttp时,直接引入包及配置以下内容根本不生效,还是走原生的。f
- 前言锁作为并发共享数据,保证一致性的工具,在JAVA平台有多种实现(如 synchronized 和 ReentrantLock等等 ) 。