二十二、状态模式

news2024/11/24 18:44:13

文章目录

  • 1 基本介绍
  • 2 案例
    • 2.1 Season 接口
    • 2.2 Spring 类
    • 2.3 Summer 类
    • 2.4 Autumn 类
    • 2.5 Winter 类
    • 2.6 Person 类
    • 2.7 Client 类
    • 2.8 Client 类的运行结果
    • 2.9 总结
  • 3 各角色之间的关系
    • 3.1 角色
      • 3.1.1 State ( 状态 )
      • 3.1.2 ConcreteState ( 具体的状态 )
      • 3.1.3 Context ( 上下文 )
      • 3.1.4 Client ( 客户端 )
    • 3.2 类图
  • 4 注意事项
  • 5 优缺点
  • 6 适用场景
  • 7 总结


1 基本介绍

状态模式(State Pattern)是一种 行为型 设计模式,它 允许一个对象在其内部状态改变时改变它的行为,使得这个对象 看起来像是修改了它的类

2 案例

本案例显示了 人 在 不同的季节 中 享受不同的假期 和 穿不同的上衣 的情况,季节的变化通过月份的变化而实现。

2.1 Season 接口

public interface Season { // 季节
    String getName(); // 季节名称
    void festivals(); // 节日情况
    void dress(); // 上衣的穿着情况
}

2.2 Spring 类

public class Spring implements Season { // 春季
	// 单例模式
    private static final Spring SPRING = new Spring();
    private Spring() {}
    public static Spring getInstance() {
        return SPRING;
    }

    @Override
    public String getName() {
        return "春季";
    }

    @Override
    public void festivals() {
        System.out.println("春节、元宵节、清明节。");
    }

    @Override
    public void dress() {
        System.out.println("从 棉袄 向 短袖 过渡。");
    }
}

2.3 Summer 类

public class Summer implements Season { // 夏季
	// 单例模式
    private static final Summer SUMMER = new Summer();
    private Summer() {}
    public static Summer getInstance() {
        return SUMMER;
    }

    @Override
    public String getName() {
        return "夏季";
    }

    @Override
    public void festivals() {
        System.out.println("端午节、劳动节。");
    }

    @Override
    public void dress() {
        System.out.println("穿短袖。");
    }
}

2.4 Autumn 类

public class Autumn implements Season { // 秋季
	// 单例模式
    private static final Autumn autumn = new Autumn();
    private Autumn() {}
    public static Autumn getInstance() {
        return autumn;
    }

    @Override
    public String getName() {
        return "秋季";
    }

    @Override
    public void festivals() {
        System.out.println("中秋节、重阳节。");
    }

    @Override
    public void dress() {
        System.out.println("从 短袖 向 棉袄 过渡。");
    }
}

2.5 Winter 类

public class Winter implements Season { // 冬季
	// 单例模式
    private static final Winter INSTANCE = new Winter();
    private Winter() {}
    public static Winter getInstance() {
        return INSTANCE;
    }

    @Override
    public String getName() {
        return "冬季";
    }

    @Override
    public void festivals() {
        System.out.println("冬至、腊八节、除夕。");
    }

    @Override
    public void dress() {
        System.out.println("穿棉袄。");
    }
}

2.6 Person 类

public class Person { // 人
    private int month; // 月份
    private Season season; // 月份对应的季节

    public void setMonth(int month) { // 设置月份和对应的季节
        if (month < 1 || month > 12) {
            throw new IllegalArgumentException("请输入正确的月份");
        }

        this.month = month;
        if (month >= 2 && month < 5) {
            season = Spring.getInstance();
        } else if (month >= 5 && month < 8) {
            season = Summer.getInstance();
        } else if (month >= 8 && month < 11) {
            season = Autumn.getInstance();
        } else {
            season = Winter.getInstance();
        }
    }

    public void showMonth() { // 显示月份和季节的情况
        System.out.println("=================="
                + month + "月是" + season.getName()
                + "==================");
    }

    public void showNotAttendClass() { // 显示不上课的情况,除了周末和寒暑假之外
        System.out.print("不上课的日子有:");
        season.festivals();
    }

    public void showDress() { // 显示上衣的穿着情况
        System.out.print("上衣的穿着情况:");
        season.dress();
    }
}

2.7 Client 类

public class Client { // 客户端,测试了 人在每个月 不上课的情况 和 上衣的穿着情况
    public static void main(String[] args) {
        Person person = new Person();
        // 每隔 1 秒增加 1 个月
        for (int month = 1; month <= 12; month++) {
            person.setMonth(month);
            person.showMonth();
            person.showNotAttendClass();
            person.showDress();
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        }
    }
}

2.8 Client 类的运行结果

==================1月是冬季==================
不上课的日子有:冬至、腊八节、除夕。
上衣的穿着情况:穿棉袄。
==================2月是春季==================
不上课的日子有:春节、元宵节、清明节。
上衣的穿着情况:从 棉袄 向 短袖 过渡。
==================3月是春季==================
不上课的日子有:春节、元宵节、清明节。
上衣的穿着情况:从 棉袄 向 短袖 过渡。
==================4月是春季==================
不上课的日子有:春节、元宵节、清明节。
上衣的穿着情况:从 棉袄 向 短袖 过渡。
==================5月是夏季==================
不上课的日子有:端午节、劳动节。
上衣的穿着情况:穿短袖。
==================6月是夏季==================
不上课的日子有:端午节、劳动节。
上衣的穿着情况:穿短袖。
==================7月是夏季==================
不上课的日子有:端午节、劳动节。
上衣的穿着情况:穿短袖。
==================8月是秋季==================
不上课的日子有:中秋节、重阳节。
上衣的穿着情况:从 短袖 向 棉袄 过渡。
==================9月是秋季==================
不上课的日子有:中秋节、重阳节。
上衣的穿着情况:从 短袖 向 棉袄 过渡。
==================10月是秋季==================
不上课的日子有:中秋节、重阳节。
上衣的穿着情况:从 短袖 向 棉袄 过渡。
==================11月是冬季==================
不上课的日子有:冬至、腊八节、除夕。
上衣的穿着情况:穿棉袄。
==================12月是冬季==================
不上课的日子有:冬至、腊八节、除夕。
上衣的穿着情况:穿棉袄。

2.9 总结

本案例让四个季节对应四个类,并且都实现了 Season 接口,从而能够通过调用 Season 的接口来动态地调用具体季节的方法,很好地利用了面向对象的 多态性,避免了很多判断分支语句,使得编写代码时更简单。

此外,由于四个具体的季节类中没有成员字段,所以使用 单例模式 来创建它们的对象实例。

3 各角色之间的关系

3.1 角色

3.1.1 State ( 状态 )

该角色负责 定义 根据不同状态进行不同处理的 接口。本案例中,Season 接口扮演了该角色。

3.1.2 ConcreteState ( 具体的状态 )

该角色负责 实现 State 角色定义的 接口。本案例中,Spring, Summer, Autumn, Winter 类都在扮演该角色。

3.1.3 Context ( 上下文 )

该角色负责 持有表示当前状态的 ConcreteState 角色的实例对象,并 定义供外部使用的接口。本案例中,Person 类扮演了该角色。

3.1.4 Client ( 客户端 )

该角色负责 使用 Context 角色完成具体的业务逻辑。本案例中,Client 类扮演了该角色。

3.2 类图

alt text
说明:

  • 有时候 State 可以使用抽象类实现,只需要注意 单继承 即可。
  • Context 表面上聚合了 State,实际上根据具体情况会聚合不同的 ConcreteState。
  • Client 可能不会直接使用 State 和 ConcreteState,而是给 Context 传递一个变量,Context 根据这个变量修改自身的状态,就像本案例一样,Client 类给 Person 类的 setMonth() 方法传递 月份 这个参数,从而修改 Person 类内部聚合的 season 对象。

4 注意事项

  • 避免过多状态:虽然状态模式可以处理多个状态,但过多的状态会使系统变得复杂,增加维护难度。因此,在设计时应尽量控制状态的数量,避免状态爆炸。
  • 遵守单一职责原则:每个 ConcreteState 应该只负责一种状态的行为,避免将多个状态的行为放在同一个类中。
  • 避免复杂逻辑:ConcreteState 中的逻辑应该尽量简单,避免在 ConcreteState 中实现复杂的业务逻辑。复杂的逻辑应该由 Context 或其他类来处理。
  • 性能考虑:虽然状态模式可以提高代码的可读性和可维护性,但在某些情况下,由于 需要频繁地创建和销毁状态对象可能会降低系统的性能。因此,应该避免不必要的对象创建和销毁,可以考虑使用 单例模式享元模式 来避免创建过多的对象。
  • 状态转换的管理:状态转换有两种管理方式,在使用本模式时需要选择具体的管理方式。
    • 在 Context 中管理
      • 优点:提高 ConcreteState 的独立性,程序的整体结构会更加清晰。
      • 缺点:Context 必须了解所有 ConcreteState。
    • 在各个 ConcreteState 中管理
      • 优点:每个 ConcreteState 都知道在什么情况下进行状态转换。
      • 缺点:每个 ConcreteState 都必须了解所有 ConcreteState。

5 优缺点

优点

  • 提高代码的可读性和可重用性:状态模式通过明确的 ConcreteState 来表示不同的状态,使得代码更加易于理解和重用。其他对象或系统也可以通过 State 与 ConcreteState 进行交互,而不需要关心具体的状态实现,从而提高了代码的 可读性可重用性
  • 增强系统的可维护性:由于每个 ConcreteState 都封装了与特定状态相关的行为,因此当需要修改某个状态的行为时,只需要修改该 ConcreteState 的代码即可,而不会影响到其他 ConcreteState 或 Context 的代码,从而增强了系统的 可维护性
  • 增强系统的扩展性:当需要添加新的状态时,只需要新增一个 ConcreteState,并在 Context 中修改状态转换的逻辑即可,而不需要修改其他 ConcreteState 的代码,这符合开闭原则(对扩展开放,对修改关闭),从而增强了系统的 扩展性

缺点

  • 增加类的数量:状态模式会引入大量的 ConcreteState,这可能会增加系统的复杂性和类的数量。当状态数量较多时,系统的维护成本也会相应增加。
  • 状态转换的逻辑可能变得复杂:在某些情况下,状态之间的转换逻辑可能非常复杂,这可能会导致 ConcreteState 的代码变得难以理解和维护。此外,如果状态转换的逻辑不恰当,还可能导致系统出现错误或不一致的行为。
  • 对开闭原则的支持有限:虽然状态模式在一定程度上支持开闭原则,但在某些情况下,添加 新的状态 或 修改状态转换的逻辑 仍然需要修改 ConcreteState 的代码,这可能会违反开闭原则,使得系统的扩展性受到一定的限制。

6 适用场景

  • 多状态行为:当 一个对象具有多种状态并且这些状态会影响其行为 时,可以使用状态模式。每个状态对应一个 ConcreteState,封装了在该状态下对象的行为。
  • 状态转换复杂:如果 对象的状态转换逻辑非常复杂并且这些转换逻辑分散在多个地方(如多个条件语句或方法中),那么使用状态模式可以将这些逻辑集中管理,使代码更加清晰和易于维护。
  • 实现状态机:状态模式是实现 状态机 的一种有效方式。状态机是 一种用于描述系统在不同状态下如何响应不同事件的模型。状态模式允许将状态机的各个状态作为独立的类来实现,并通过状态转换来模拟状态机的行为。

7 总结

状态模式 是一种 行为型 设计模式,它使用 多态性 来区分对象在不同状态下的不同行为,避免了在多处书写多重分支语句的复杂性。在本模式中,对象的 状态转换 是一个重点,使用本模式之前需要明确对象的状态应该在合适变化。本模式在生产中比较常用,因为现实生活中有很多实体需要有状态的概念,例如用户的标签。

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

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

相关文章

【ARM+Codesys 客户案例 】 基于RK3568/A40i/STM32+CODESYS开发AGV运动控制器,支持国产定制

在过去&#xff0c;步科更多的是为AGV客户提供单一、高性能的低压伺服核心部件产品&#xff0c;而现在&#xff0c;步科基于 CODESYS 开发了一款面向AGV机器人的特种控制器 - 青龙1号&#xff0c;开始提供以步科AGV运动控制器FD1X4S系列低压伺服Green系列HMI等为核心的AGV总线控…

keepalived理论--实验

一 . 高可用集群 1.1 集群类型 LB &#xff1a; Load Balance 负载均衡 LVS/HAProxy/nginx &#xff08; http/upstream, stream/upstream &#xff09; HA &#xff1a; High Availability 高可用集群 数据库、 Redis SPoF: Single Point of Failure &#xff0c;解决…

2004-2023华为杯数学建模优秀参考论文

笔者整理了2004-2023年华为杯研究生数学建模全部优秀论文和赛题&#xff0c;内容齐全&#xff0c;适合将要参加建模比赛的朋友学习使用。 免费优秀论文获取联系&#xff1a; 建模忠哥小师妹 2004-2023历届华为杯研究生数学建模优秀论文合集&#xff1a;

【数学分析笔记】第2章第1节实数系的连续性(2)

2. 数列极限 2.1 实数系的连续性 2.1.3 确界存在定理 【定理2.1.1】&#xff08;确界存在定理——实数系连续性定理&#xff09;非空有上界的数集必有上确界&#xff0c;非空有下界的数集必有下确界。 【证】&#xff08;写了一些我自己的理解&#xff0c;欢迎数院大神批评指…

Linux基础知识学习(二)

一. 常用基本命令 1. 目录管理 1> 绝对路径、相对路径 绝对路径路径的全称&#xff1a;C:\ProgramData\360safe\xxx.xx 比如说 360safe 目录下&#xff0c;那这个 xxx.xx 文件&#xff0c;对应我们的相对配置就 /xxx.xx cd &#xff1a; 切换目录命令&#xff01; ./ &…

【html+css 绚丽Loading】-000001 双极乾坤盘

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享htmlcss 绚丽Loading&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495…

【信创】麒麟打包工具初体验

往期好文&#xff1a;关于信创系统&#xff08;麒麟、统信、中科方德&#xff09;的10个问题与答复&#xff08;二&#xff09; Hello&#xff0c;大家好啊&#xff01;今天给大家带来一篇关于麒麟桌面操作系统上麒麟打包工具的介绍与使用的文章。麒麟打包工具是一款专门为麒麟…

C/C++ 不定参函数

C语言不定参函数 函数用法总结 Va_list 作用&#xff1a;类型定义&#xff0c;生命一个变量&#xff0c;该变量被用来访问传递给不定参函数的可变参数列表用法&#xff1a;供后续函数进调用&#xff0c;通过该变量访问参数列表 typedefchar* va_list; va_start 作用&#xff…

解决MSPM0G3507芯片锁住的问题

编译环境&#xff1a;Windows 开发软件&#xff1a;Keil 开发主控&#xff1a;立创的MSPM0G3507 我们在MSPM0G3507时&#xff0c;常为芯片锁住烦恼&#xff0c;常见的锁死是因为使用了ST-Link&#xff0c;这里展示的是使用ST-Link后芯片锁死的解决步骤。 现象&a…

MySQL数据库入门,pycharm连接数据库—详细讲解

一.安装MySQL 1.常用MySQL5.7&#xff0c;首先安装MySQL&#xff0c; &#xff08;一&#xff09; &#xff08;二&#xff09; &#xff08;三&#xff09; &#xff08;四&#xff09; &#xff08;五&#xff09; 2.配置环境变量 打开MySQL安装路径&#xff0c;在其中找到…

python:画函数积分图

《高等数学》同济大学版 P209 编写 test_diff_area.py 如下 # -*- coding: utf-8 -*- """ 函数积分图 y x^3 -x^2 -x1 """ import numpy as np import matplotlib.pyplot as plt from matplotlib.patches import Polygondef func(x):return …

人工智能的普及造成了一场能源危机,但并不是无法解决。

人工智能的普及造成了一场能源危机&#xff0c;但并不是无法解决。 原文链接&#xff1a; 点击访问我的技术博客https://ai.weoknow.comhttps://ai.weoknow.com 人工智能&#xff08;AI&#xff09;的爆炸式增长促使科技巨头&#xff08;包括 Google、Meta、亚马逊和微软等&…

LeetCode刷题笔记第231题:2 的幂

LeetCode刷题笔记第231题&#xff1a;2 的幂 题目&#xff1a; 想法&#xff1a; 对输入的数值循环除以2直至数值小于等于1&#xff0c;如果最终的数值为1则为2的幂&#xff0c;小于1则不是2的幂。 class Solution:def isPowerOfTwo(self, n: int) -> bool:if n 1:retur…

[C#]winform基于opencvsharp结合Diffusion-Low-Light算法实现低光图像增强黑暗图片变亮变清晰

【训练源码】 https://github.com/JianghaiSCU/Diffusion-Low-Light 【参考源码】 https://github.com/hpc203/Diffusion-Low-Light-onnxrun 【论文地址】 https://arxiv.org/pdf/2306.00306.pdf 【算法原理图】 【效果展示】 【测试环境】 vs2019 netframework4.7.2 …

【数据结构】关于优先级队列(堆),你了解内部原理吗?(超详解!!!)

前言&#xff1a; &#x1f31f;&#x1f31f;Hello家人们&#xff0c;这期讲解二叉树的遍历&#xff0c;希望你能帮到屏幕前的你。 &#x1f308;上期博客在这里&#xff1a;http://t.csdnimg.cn/EdeWV &#x1f308;感兴趣的小伙伴看一看小编主页&#xff1a;GGBondlctrl-CSD…

Iinux脚本bash:对自己的应用程序及其相关目录进行备份和恢复,并可查看备份计划、备份状态、备份大小等

目录 一、要求 1、需求 2、需求分析 二、脚本 1、总述 2、创建备份脚本 &#xff08;1&#xff09;脚本命名 &#xff08;2&#xff09;脚本内容 3、创建恢复脚本 &#xff08;1&#xff09;脚本命名 &#xff08;2&#xff09;脚本内容 4、设置cron作业 5、监控脚…

mybatis、mybatis-plus自定义插件,实现自定义策略数据脱敏功能

背景 mybatis中四大组件的作用,下面开发的插件拦截器会使用 四大组件Executor、StatementHandler、ParameterHandler、ResultSetHandler 需求 1、根据脱敏规则进行查询数据,显示的时候进行展示脱敏 2、根据脱敏规则进行查询数据,将脱敏后的数据批量更新回数据库,进行脱…

【Python系列】 Python打印99乘法表

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

Vue+ElementUI技巧分享:创建一个带有进度显示的文件下载和打包组件

在现代前端开发中&#xff0c;用户体验至关重要&#xff0c;尤其是在处理文件下载时。为用户提供实时的下载进度显示和打包功能&#xff0c;不仅能提升用户体验&#xff0c;还能使应用更具专业性。在本文中&#xff0c;我们将创建一个 Vue 组件&#xff0c;用于显示文件下载进度…

视图变化 - 等比例变换防止视图拉伸

文章目录 使用场景等比变换等高填充等宽填充代码进行比目标宽高计算超出部分处理设置负的 marginclip 裁剪 End参考&#xff1a; 使用场景 在日常开发中&#xff0c;经常会遇到的一个需求是将图片/视频从界面的一个位置&#xff0c;变换到另一个位置。在处理这类问题的时候经常…