概念:

  State模式也叫状态模式,是行为设计模式的一种。State模式允许通过改变对象的内部状态而改变对象的行为,这个对象表现得就好像修改了它的类一样。

  根据这个概念,我们举个例子

  1. public class Behavior {
  2. private int time;
  3.  
  4. public int getTime() {
  5. return time;
  6. }
  7.  
  8. public void setTime(int time) {
  9. this.time = time;
  10. }
  11.  
  12. public void eat(){
  13. if(time == 7){
  14. System.out.println("吃早饭");
  15. }else if(time == 12){
  16. System.out.println("吃午饭");
  17. }else if(time == 18){
  18. System.out.println("吃晚饭");
  19. }else{
  20. System.out.println("还不到吃饭时间");
  21. }
  22. }
  23. }
  1. public class MainClass {
  2. public static void main(String[] args) {
  3. Behavior behavior = new Behavior();
  4. behavior.setTime(7);
  5. behavior.eat();
  6.  
  7. behavior.setTime(12);
  8. behavior.eat();
  9.  
  10. behavior.setTime(18);
  11. behavior.eat();
  12.  
  13. behavior.setTime(20);
  14. behavior.eat();
  15. }
  16. }

  结果:

  可以看到,根据time属性的不同,对象的行为也发生了改变,但是这样的方式很不好,所有的事情都放到了eat()方法中,导致eat()方法过于复杂

  下面就看一看状态模式

  状态模式的应用场景

  状态模式主要解决的是当控制一个对象状态转换的条件表达式过于复杂时的情况。把状态的判断逻辑转译到表现不同状态的一系列类当中,可以把复杂的判断逻辑简化。

  简单来说:

  1.一个对象的行为取决于它的状态,并且它必须在运行时刻根据状态改变它的行为。
  2.一个操作中含有庞大的多分支结构,并且这些分支决定于对象的状态。

  状态模式的结构

  

  状态模式的角色和职责

  1、Context:用户对象:拥有一个State类型的成员,以标识对象的当前状态(Behavior)

  2、State:接口或基类封装与Context的特定状态相关的行为;

  3、ConcreteState:接口实现类或子类实现了一个与Context某个状态相关的行为。

  按照状态模式,我们来改造一下,刚才的例子,吃早中晚饭,不是吃饭时间,都是状态,所以我们把状态单独封装出来。

  首先,新建一个State

  1. public abstract class State {
  2. public abstract void eat();
  3. }

  接着新建ConcreteState

  1. /*
  2. * 早餐
  3. */
  4. public class BreakfastState extends State {
  5.  
  6. @Override
  7. public void eat() {
  8. System.out.println("吃早餐");
  9. }
  10.  
  11. }
  1. /*
  2. * 午餐
  3. */
  4. public class LunchState extends State {
  5.  
  6. @Override
  7. public void eat() {
  8. System.out.println("吃午餐");
  9. }
  10.  
  11. }
  1. /*
  2. * 晚餐
  3. */
  4. public class DinnerState extends State {
  5.  
  6. @Override
  7. public void eat() {
  8. System.out.println("吃晚餐");
  9. }
  10.  
  11. }
  1. /*
  2. * 不是吃饭时间
  3. */
  4. public class NoFoodState extends State {
  5.  
  6. @Override
  7. public void eat() {
  8. System.out.println("不是吃饭时间");
  9. }
  10.  
  11. }

  再修改一下behavior

  1. public class Behavior {
  2. private int time;
  3. State state = null;
  4.  
  5. public int getTime() {
  6. return time;
  7. }
  8.  
  9. public void setTime(int time) {
  10. this.time = time;
  11. }
  12.  
  13. public void eat(){
  14. if(time == 7){
  15. state = new BreakfastState();
  16. state.eat();
  17. }else if(time == 12){
  18. state = new LunchState();
  19. state.eat();
  20. }else if(time == 18){
  21. state = new DinnerState();
  22. state.eat();
  23. }else{
  24. state = new NoFoodState();
  25. state.eat();
  26. }
  27. }
  28. }

  这样,和刚才的结果一样,但是这样子,判断逻辑还是在对象中,我们继续修改,将逻辑写到ConcreteState

  因为,我们要知道time,所以需要向state中传入参数,所以我们将Behavior传进去

  1. public abstract class State {
  2. public abstract void eat(Behavior behavior);
  3. }

  然后,修改Behavior

  1. public class Behavior {
  2. private int time;
  3. State state = null;
  4.  
  5. public Behavior() {
  6. state = new BreakfastState();
  7. }
  8.  
  9. public int getTime() {
  10. return time;
  11. }
  12.  
  13. public void setTime(int time) {
  14. this.time = time;
  15. }
  16.  
  17. public State getState() {
  18. return state;
  19. }
  20.  
  21. public void setState(State state) {
  22. this.state = state;
  23. }
  24.  
  25. public void eat(){
  26. //逻辑取出,所以这里只剩调用方法
  27. state.eat(this);
  28. //当所有方法都完成后,回到最初始状态
  29. state = new BreakfastState();
  30. }
  31. }

  接着,再继续修改每一个ConcreteState

  1. /*
  2. * 早餐
  3. */
  4. public class BreakfastState extends State {
  5.  
  6. @Override
  7. public void eat(Behavior behavior) {
  8. if(behavior.getTime() == 7){
  9. System.out.println("吃早餐");
  10. }else{
  11. //如果不符合条件,重置state为下一个状态
  12. behavior.setState(new LunchState());
  13. behavior.eat();
  14. }
  15. }
  16. }
  1. /*
  2. * 午餐
  3. */
  4. public class LunchState extends State {
  5.  
  6. @Override
  7. public void eat(Behavior behavior) {
  8. if(behavior.getTime() == 12){
  9. System.out.println("吃午餐");
  10. }else{
  11. behavior.setState(new DinnerState());
  12. behavior.eat();
  13. }
  14. }
  15.  
  16. }
  1. /*
  2. * 晚餐
  3. */
  4. public class DinnerState extends State {
  5.  
  6. @Override
  7. public void eat(Behavior behavior) {
  8. if(behavior.getTime() == 18){
  9. System.out.println("吃晚餐");
  10. }else{
  11. behavior.setState(new NoFoodState());
  12. behavior.eat();
  13. }
  14. }
  15.  
  16. }
  1. /*
  2. * 不是吃饭时间
  3. */
  4. public class NoFoodState extends State {
  5.  
  6. @Override
  7. public void eat(Behavior behavior) {
  8. System.out.println("不是吃饭时间");
  9. }
  10.  
  11. }

  这样,结果和之前是一样的

  状态模式的优点与缺点

  优点: 1、封装了转换规则。

      2、枚举可能的状态,在枚举状态之前需要确定状态种类。

      3、将所有与某个状态有关的行为放到一个类中,并且可以方便地增加新的状态,只需要改变对象状态即可改变对象的行为。

      4、允许状态转换逻辑与状态对象合成一体,而不是某一个巨大的条件语句块。

      5、可以让多个环境对象共享一个状态对象,从而减少系统中对象的个数。

  缺点: 1、状态模式的使用必然会增加系统类和对象的个数。

      2、状态模式的结构与实现都较为复杂,如果使用不当将导致程序结构和代码的混乱。

        3、状态模式对"开闭原则"的支持并不太好,对于可以切换状态的状态模式,增加新的状态类需要修改那些负责状态转换的源代码,否则无法切换到新增状态,而且修改某个状态类的行为也需修改对应类的源代码。

 注意事项:在行为受状态约束的时候使用状态模式,而且状态不超过 5 个。

java设计模式-----22、状态模式的更多相关文章

  1. 折腾Java设计模式之状态模式

    原文地址 折腾Java设计模式之状态模式 状态模式 在状态模式(State Pattern)中,类的行为是基于它的状态改变的.这种类型的设计模式属于行为型模式.在状态模式中,我们创建表示各种状态的对象 ...

  2. JAVA设计模式--State(状态模式)

    状态模式(State Pattern)是设计模式的一种,属于行为模式. 定义(源于Design Pattern):当一个对象的内在状态改变时允许改变其行为,这个对象看起来像是改变了其类. 状态模式主要 ...

  3. JAVA设计模式:状态模式

    声明:转载请说明来源:http://www.cnblogs.com/pony1223/p/7518226.html 一.引出状态模式 假设我们现在有一个糖果机项目,那么我们知道正常一般糖果机提供给用户 ...

  4. Java设计模式之状态模式详解

    (本文由言念小文原创,转载请注明出处) 在实际工作中经常遇到某个对象,处于不同的状态有不同行为逻辑.且状态之间可以相互迁移的业务场景,特别是在开发通信协议栈类软件中尤为多见.<设计模式之禅> ...

  5. java设计模式之状态模式

    状态模式 允许一个对象在其内部状态改变时改变它的行为,对象看起来似乎修改了它的类. 状态模式UML图 上下文环境(Context):它定义了客户程序需要的接口并维护一个具体状态角色的实例,将与状态相关 ...

  6. 曾侯乙编钟引发的遐想之Java设计模式:状态模式

    目录 示例 简单例子 改进代码 状态模式 定义 意图 主要解决问题 何时使用 优缺点 曾侯乙编钟 状态模式-命令模式-策略模式 示例 一个类对外提供了多个行为,同时该类对象有多种状态,不同状态下对外的 ...

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

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

  8. 折腾Java设计模式之备忘录模式

    原文地址:折腾Java设计模式之备忘录模式 备忘录模式 Without violating encapsulation, capture and externalize an object's int ...

  9. 折腾Java设计模式之访问者模式

    博客原文地址:折腾Java设计模式之访问者模式 访问者模式 Represent an operation to be performed on the elements of an object st ...

  10. 折腾Java设计模式之命令模式

    博客原文地址 折腾Java设计模式之命令模式 命令模式 wiki上的描述 Encapsulate a request as an object, thereby allowing for the pa ...

随机推荐

  1. 【BZOJ3709】 [PA2014]Bohater(贪心)

    传送门 BZOJ Solution 考虑如果可以回血肯定要打,那么就是按照伤害值从小到大排个序能打就打,不能打就\(NIE\). 接着看不能够回血的,emmm,把这个过程反着看一下就是打一个怪扣\(a ...

  2. 【062新题】OCP 12c 062出现大量新题-15

    choose one In your Oracle 12c database, you plan to execute the command: SQL> CREATE TABLESPACE t ...

  3. JFrame、JPanel 、Layout开发的简单例子

    写了Java这么久,居然发现想手写一个带网格袋布局的JFrame,还不记得怎么写,写了这么多代码真不敢说记得所有细节. 幸好,只要记清楚概念就能快速开发.首先,明确一下3种容器类的差别和用途: No. ...

  4. python iter函数用法

    iter函数用法简述 Python 3中关于iter(object[, sentinel)]方法有两个参数. 使用iter(object)这种形式比较常见. iter(object, sentinel ...

  5. odoo 开发基础 -- 视图之widget

    Odoo 中的widget many2many_tags one2many_list selection progressbar selection statusbar handle monetary ...

  6. 利用Warensoft Stock Service编写高频交易软件--DEMO

    利用Warensoft Stock Service编写高频交易软件 无论是哪种交易软件,对于程序员来讲,最麻烦的就是去实现各种算法.本文以SAR算法的实现过程为例,为大家说明如何使用Warensoft ...

  7. EFCore.MySql当模型遇到int[]怎么办

    我使用的是Pomole.EntityFrameworkCore.MySql 需要将旧项目中的excels表转成实体,其中有一列是json格式的int[] 当遇到第一张表的时候,我使用了这样的方法来读取 ...

  8. Facade外观模式(结构性模式)

    1.系统的复杂度 需求:开发一个坦克模拟系统用于模拟坦克车在各种作战环境中的行为,其中坦克系统由引擎.控制器.车轮等各子系统构成.然后由对应的子系统调用. 常规的设计如下: #region 坦克系统组 ...

  9. RandomStringUtils工具类

    //产生5位长度的随机字符串,中文环境下是乱码 RandomStringUtils.random(5); //使用指定的字符生成5位长度的随机字符串 RandomStringUtils.random( ...

  10. JVM Scan

    1.jmap -histo pid|head -100 2.jstat -gcutil pid cycle 3.jmap -heap pid