概述

描述

  • 允许一个对象在其内部状态改变时改变它的行为,对象看起来似乎修改了它的类。

  • 其别名为状态对象(Objects for States),状态模式是一种对象行为型模式。

  • 有限状态机(FSMs)

    • 拥有状态机所有可能状态的集合
    • 状态机同时只能在一个状态
    • 一连串的输入或事件被发送给状态机
    • 每个状态都有一系列的转移,每个转移与输入和另一状态相关
    • 动画状态机系统、行为树系统来理解
  • 并发状态机

    • 有些状态需要并行执行,例如动画状态机,经常分为上半身动画与下半身动画融合,如装备动作、射击动作、换弹动作与行走动作并行
  • 层次状态机

    • 状态中嵌套子状态,可以使用继承实现或者状态栈来实现
  • 下推自动机

    • 用栈来存储一系列状态。有限状态机有一个指向状态的指针,下推自动机有一栈指针
    • 新状态压入栈中,“当前的”状态总是在栈顶
    • 弹出最上面的状态,这个状态会被销毁,它下面的状态成为新状态。如UI界面管理

套路

  • 环境类 Context,用于改变状态
  • 抽象状态类 State
  • 具体状态类 ConcreteState

使用场景

  • 对象的行为依赖于它的状态(属性)并且可以根据它的状态改变而改变它的相关行为。
  • 代码中包含大量与对象状态有关的条件语句,这些条件语句的出现,会导致代码的可维护性和灵活性变差,不能方便地增加和删除状态,使客户类与类库之间的耦合增强。在这些条件语句中包含了对象的行为,而且这些条件对应于对象的各种状态。
  • 示例
    • OA办公系统中多种状态:尚未办理;正在办理;正在批示;正在审核
    • TCP 连接状态
    • 动画系统、AI行为树
    • UI界面管理

优缺点

  • 优点

    • 封装了转换规则。
    • 枚举可能的状态,在枚举状态之前需要确定状态种类。
    • 将所有与某个状态有关的行为放到一个类中,并且可以方便地增加新的状态,只需要改变对象状态即可改变对象的行为。
    • 允许状态转换逻辑与状态对象合成一体,而不是某一个巨大的条件语句块。
    • 可以让多个环境对象共享一个状态对象,从而减少系统中对象的个数。
  • 缺点
    • 状态模式的使用必然会增加系统类和对象的个数。
    • 状态模式的结构与实现都较为复杂,如果使用不当将导致程序结构和代码的混乱。
    • 状态模式对“开闭原则”的支持并不太好,对于可以切换状态的状态模式,增加新的状态类需要修改那些负责状态转换的源代码,否则无法切换到新增状态;而且修改某个状态类的行为也需修改对应类的源代码。

UE4 实践

  • 在UE4中,实现切换UI状态

  • 创建状态抽象类,以接口的形式

    UINTERFACE(MinimalAPI)
    class UStateInterface : public UInterface
    {
    GENERATED_BODY()
    }; /**
    *
    */
    class DESIGNPATTERNS_API IStateInterface
    {
    GENERATED_BODY() // Add interface functions to this class. This is the class that will be inherited to implement this interface.
    public:
    virtual void EnterState() = 0;
    virtual void ExitState() = 0;
    };
  • 创建状态具体类UBaseStateWidget ,作为UI的父类,切换时使用

    UCLASS()
    class DESIGNPATTERNS_API UBaseStateWidget : public UUserWidget, public IStateInterface
    {
    GENERATED_BODY()
    public:
    virtual void EnterState() override;
    virtual void ExitState() override; UFUNCTION(BlueprintNativeEvent,BlueprintCallable,Category="State Pattern")
    void OnEnterState();
    UFUNCTION(BlueprintNativeEvent, BlueprintCallable, Category = "State Pattern")
    void OnExitState();
    };
    #include "BaseStateWidget.h"
    
    void UBaseStateWidget::EnterState()
    {
    OnEnterState();
    } void UBaseStateWidget::ExitState()
    {
    OnExitState();
    } void UBaseStateWidget::OnEnterState_Implementation()
    {
    AddToViewport();
    } void UBaseStateWidget::OnExitState_Implementation()
    {
    RemoveFromParent();
    }
  • 创建状态管理类

    UCLASS()
    class DESIGNPATTERNS_API AUIStateManager : public AActor
    {
    GENERATED_BODY() public:
    // Sets default values for this actor's properties
    AUIStateManager(); protected:
    // Called when the game starts or when spawned
    virtual void BeginPlay() override; public:
    // Called every frame
    virtual void Tick(float DeltaTime) override; // 改变状态
    UFUNCTION(BlueprintCallable, Category = "State Pattern")
    void EnterState(TSubclassOf<UBaseStateWidget> StateWidgetClass); // 退出所有状态
    UFUNCTION(BlueprintCallable, Category = "State Pattern")
    void ExitAllState(); // 当前状态实例
    UPROPERTY(BlueprintReadWrite, Category = "State Pattern")
    UBaseStateWidget* CurrentStateWidget; private:
    // 存储状态实例
    TMap<TSubclassOf<UBaseStateWidget>, UBaseStateWidget*> WidgetInstances;
    };
    void AUIStateManager::EnterState(TSubclassOf<UBaseStateWidget> StateWidgetClass)
    {
    if (CurrentStateWidget != nullptr)
    {
    CurrentStateWidget->ExitState();
    } if (WidgetInstances.Contains(StateWidgetClass))
    {
    CurrentStateWidget = WidgetInstances.FindRef(StateWidgetClass);
    }
    else
    {
    APlayerController* PC = UGameplayStatics::GetPlayerController(GetWorld(), 0);
    CurrentStateWidget = CreateWidget<UBaseStateWidget>(PC, StateWidgetClass);
    WidgetInstances.Add(StateWidgetClass,CurrentStateWidget);
    }
    CurrentStateWidget->EnterState();
    } void AUIStateManager::ExitAllState()
    {
    for (auto& Elem : WidgetInstances)
    {
    (Elem.Value)->ExitState();
    }
    }
  • 创建状态类UBaseStateWidget 的蓝图派生类

    • WBP_Start —— 开始界面
    • WBP_LoadingScreen —— 加载界面
    • WBP_Save —— 存档界面
    • WBP_Option —— 选项界面



  • 创建 UIStateManager 蓝图派生类

  • 效果

  • 扩展,可进一步用栈状态机实现界面管理


参考

【UE4 设计模式】状态模式 State Pattern的更多相关文章

  1. C#设计模式——状态模式(State Pattern)

    一.概述在面向对象软件设计时,常常碰到某一个对象由于状态的不同而有不同的行为.如果用if else或是switch case等方法处理,对象操作及对象的状态就耦合在一起,碰到复杂的情况就会造成代码结构 ...

  2. 乐在其中设计模式(C#) - 状态模式(State Pattern)

    原文:乐在其中设计模式(C#) - 状态模式(State Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 状态模式(State Pattern) 作者:webabcd 介绍 允 ...

  3. 二十四种设计模式:状态模式(State Pattern)

    状态模式(State Pattern) 介绍允许一个对象在其内部状态改变时改变它的行为.对象看起来似乎修改了它所属的类. 示例有一个Message实体类,对它的操作有Insert()和Get()方法, ...

  4. 状态模式-State Pattern(Java实现)

    状态模式-State Pattern 在状态模式(State Pattern)中,类的行为是基于它的状态改变的.当一个对象的内在状态改变时允许改变其行为,这个对象看起来像是改变了其类. State接口 ...

  5. 设计模式(十二):通过ATM取款机来认识“状态模式”(State Pattern)

    说到状态模式,如果你看过之前发布的重构系列的文章中的<代码重构(六):代码重构完整案例>这篇博客的话,那么你应该对“状态模式”并不陌生,因为我们之前使用到了状态模式进行重构.上一篇博客我们 ...

  6. [设计模式] 20 状态模式 State Pattern

    在GOF的<设计模式:可复用面向对象软件的基础>一书中对状态模式是这样说的:允许一个对象在其内部状态改变时改变它的行为.对象看起来似乎修改了它的类.状态模式的重点在于状态转换,很多时候,对 ...

  7. 十一个行为模式之状态模式(State Pattern)

    定义: 当一个对象有多个状态,并且在每个状态下有不同的行为,可以使用状态模式来在其内部改变状态时改变其行为,而客户端不会察觉状态的改变,仍使用同样的方法或接口与对象进行交互. 结构图: Context ...

  8. 状态模式(State Pattern)

    当一个对象的内在状态改变时允许改变其行为,这个对象看起来像是改变了其类. 状态模式主要解决的是当控制一个对象状态的条件表达式过于复杂时的情况.把状态的判断逻辑转移到表示不同状态的一系列类中,可以把复杂 ...

  9. 大话设计模式--状态模式 State -- C++实现实例

    1.状态模式: 当一个对象的内在状态改变时,允许改变其行为,这个对象看起来就像是改变了其类. 状态模式解决的是当控制一个对象状态转换的条件表达式过于复杂, 把状态的判断逻辑转移到表示不同状态的一系列类 ...

随机推荐

  1. Appium自动化(16) - 使用手机浏览器进行自动化测试

    如果你还想从头学起Appium,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1693896.html 前言 前面我都讲的都是针对 app ...

  2. Linux常用命令 - nl命令详解

    21篇测试必备的Linux常用命令,每天敲一篇,每次敲三遍,每月一循环,全都可记住!! https://www.cnblogs.com/poloyy/category/1672457.html 显示行 ...

  3. python基础--网站推荐

    Python教程 - 廖雪峰的官方网站 Python 基础教程 | 菜鸟教程 随笔分类 - 机器学习

  4. 【第二十篇】-Maven IntelliJ之Spring Cloud直播商城 b2b2c电子商务技术总结

    Maven IntelliJ IntelliJ IDEA 已经内建了对 Maven 的支持.我们在此例中使用的是 IntelliJ IDEA 社区版 11.1. IntelliJ IDEA 的一些特性 ...

  5. 'Specifying a namespace in include() without providing an app_name '报错解决

    需要在每个ap下面的url.py 加入一个指定app的名字 比如  user  app  下的 url.py  文件加入: urlpatterns = []app_name = "user& ...

  6. 我爬取交通学博士付费的GIS资源,每年被动收入2w很简单?

    目录 1.背景介绍 2.技术路线 3.数据结果 4.数据分析 5.总结 6.后记 1.背景介绍 某周末闲来无事,顺手打开了CSDN,看到了一个人发布的收费GIS资源,售价是¥19.9,POI数据也有人 ...

  7. PKI及SSL协议分析PKI及SSL协议分析

    任务一:搭建CA服务器 本任务初步了解CA服务器的原理和配置过程.操作都在CA服务器上. 1.远程桌面方式登录到CA服务器,在CMD下查看本机IP地址: 2.安装证书服务 依次点击:"开始& ...

  8. 一文让你快速入门pytest框架

    pytest是什么 官方文档描述: pytest is a framework that makes building simple and scalable tests easy. Tests ar ...

  9. FastAPI logger日志记录方案 loguru模块

    实现方式: 采用 loguru 模块.跟flask直接挂载到app上有区别,当然也可以尝试去这样做. 但是 好像没有这个必要.要的就是个快速.整那些子虚乌有的东西完全木有意义. 1.首先是去项目git ...

  10. VMware虚拟机常见问题(针对目前我所学的而言,还会不断更新)

    VMware虚拟机常见问题(针对目前我所学的而言,还会不断更新) 自己电脑的telnet Client是否打开 在控制面板->程序->打开或关闭Windows功能 虚拟机的telnet是否 ...