C#学习笔记之状态模式详解

本文通过例题为大家讲解C#学习笔记之状态模式,供大家参考,具体内容如下

题目1:通过代码描述每一天的不同时间段的工作效率

分析:

  首先确定,工作状态指正在工作,或者以及下班这些情况,而这些情况所受影响的因素包括:当前时间以及任务是否已经完成。所以在Work中需要两个属性:hour和finish。然后根据这两个属性直接判断当前的工作状态即可。

实现:

class Program
  {
    static void Main(string[] args)
    {
      //紧急项目
      Work emergencyProject = new Work();
      emergencyProject.Hour = 9;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 10;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 12;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 13;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 14;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 17;
      emergencyProject.Finish = false;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 19;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 22;
      emergencyProject.WriteProgram();

      Console.Read();
    }
    public class Work
    {
      private int hour;
      private bool finish = false;

      public int Hour
      {
        get { return hour; }
        set { hour = value; }
      }
      public bool Finish
      {
        get { return finish; }
        set { finish = value; }
      }
      /// <summary>
      /// 根据时间来判断当前的工作状态如何
      ///
      /// 方法过长
      /// </summary>
      public void WriteProgram()
      {
        if (hour < 12)
        {
          Console.WriteLine("当前时间:{0}点 上午工作,精神百倍", hour);
        }
        else if (hour < 13)
        {
          Console.WriteLine("当前时间:{0}点 饿了,午饭;犯困,午休", hour);
        }
        else if (hour < 17)
        {
          Console.WriteLine("当前时间:{0}点 下午状态还不错,继续努力", hour);
        }
        else
        {
          if (finish)
          {
            Console.WriteLine("当前时间:{0}点 下班回家了", hour);
          }
          else
          {
            if (hour < 21)
            {
              Console.WriteLine("当前时间:{0}点 加班哦,疲惫至极", hour);
            }
            else
            {
              Console.WriteLine("当前时间:{0}点 不行了,睡着了。", hour);
            }
          }
        }
      }
    }
  }

题目2:从1中可以很明显的看出WriteProgram()函数几乎承载了所有的判断处理,且函数函数过长,所以在Work类中违背了“单一职责原则"。

分析:

  面向对象设计就是希望做到代码的责任分解,在这个情况中,我们将所有的判断全部添加在了WriteProgram()一个函数中,不仅是函数冗长,如果我临时添加新的情况,可能对原来判断有出入的时间也会造成很多麻烦,对于将来的修改百害而无一利。

  所以我们需要根据情况来分解函数和类。

  本题目中,我们需要获得当前的工作状态,而这个的直接影响因素包括时间Hour和是否已经完成任务TaskFinished。

  所以我们根据当前的时间可以很明显的分为不同时间段,也就是WriteProgram()的if,else都可以自己单独拥有一个类,他们都继承于同一个抽象类State即可。

  然后我们初始化直接进入ForenoonState类,毕竟每天早上的时间不会变,如果当前时间不再是早上,那么我们就需要进入下一个时间段进行判断,所以在ForenoonState类中我们需要加入判断,如果当前时间满足早上的时间段,显示工作状态,如果当前时间不再是早上的时间段,那么我们进入下一个时间段,以此类推,找到我们当前时间段的工作状态。

实现:

class Program
  {
    static void Main(string[] args)
    {
      //紧急项目
      Work emergencyProject = new Work();
      emergencyProject.Hour = 9;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 10;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 12;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 13;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 14;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 17;
      emergencyProject.TaskFinished = false;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 19;
      emergencyProject.WriteProgram();
      emergencyProject.Hour = 22;
      emergencyProject.WriteProgram();

      Console.Read();
    }
    public class Work
    {
      private int hour;
      private bool finish = false;
      private State current;
      //初始化进入的时候是上午工作状态
      public Work()
      {
        current = new ForenoonState();
      }
      public int Hour
      {
        get { return hour; }
        set { hour = value; }
      }
      public bool TaskFinished
      {
        get { return finish; }
        set { finish = value; }
      }

      public void SetState(State s)
      {
        current = s;
      }
      public void WriteProgram()
      {
        current.WriteProgram(this);
      }
    }
    /// <summary>
    /// 抽象状态
    /// </summary>
    public abstract class State
    {
      public abstract void WriteProgram(Work w);
    }
    /// <summary>
    /// 上午工作状态
    ///
    /// 逻辑是:根据时间推移,上午工作后,移至中午午饭时间
    /// </summary>
    public class ForenoonState : State
    {
      public override void WriteProgram(Work w)
      {
        if (w.Hour < 12)
        {
          Console.WriteLine("当前时间:{0} 上午工作,精神百倍", w.Hour);
        }
        else
        {
          w.SetState(new NoonState());
          w.WriteProgram();
        }
      }
    }
    /// <summary>
    /// 中午工作状态
    ///
    /// 逻辑是:根据时间推移,午休时间后,移至下午工作时间
    /// </summary>
    public class NoonState : State
    {
      public override void WriteProgram(Work w)
      {
        if (w.Hour < 13)
        {
          Console.WriteLine("当前时间:{0} 饿了,午饭;犯困,午休", w.Hour);
        }
        else
        {
          w.SetState(new AfternoonState());
          w.WriteProgram();
        }
      }
    }
    /// <summary>
    /// 下午工作状态
    ///
    /// 逻辑是:根据时间推移,下午工作后,移至晚间工作状态
    /// </summary>
    public class AfternoonState : State
    {
      public override void WriteProgram(Work w)
      {
        if (w.Hour < 17)
        {
          Console.WriteLine("当前时间:{0} 下午状态还不错,继续努力", w.Hour);
        }
        else
        {
          w.SetState(new EveningState());
          w.WriteProgram();
        }
      }
    }
    /// <summary>
    /// 晚间工作状态
    ///
    /// 逻辑是:已完成工作的可以准点下班,移至下班休息状态
    ///     未完成工作的继续加班:这里又分为未超过人体承受能力的(21小时)就继续加班(此处已是最底层,没有再多加其他的操作了)
    ///                超过21小时的将移至睡眠状态
    /// </summary>
    public class EveningState : State
    {
      public override void WriteProgram(Work w)
      {
        if (w.TaskFinished)
        {
          w.SetState(new RestState());
          w.WriteProgram();
        }
        else
        {
          //最底层,不要再次调用,会无限往复的
          if (w.Hour < 21)
          {
            Console.WriteLine("当前时间:{0} 加班哦,疲惫至极", w.Hour);
          }
          else
          {
            w.SetState(new SleepingState());
            w.WriteProgram();
          }
        }
      }
    }
    /// <summary>
    /// 睡眠状态
    ///
    /// 最底层,不要再次调用,会无限往复的
    /// </summary>
    public class SleepingState : State
    {
      public override void WriteProgram(Work w)
      {
        Console.WriteLine("当前时间:{0}点 不行了,睡着了", w.Hour);
      }
    }
    /// <summary>
    /// 下班休息状态
    ///
    /// 最底层,不要再次调用,会无限往复的
    /// </summary>
    public class RestState : State
    {
      public override void WriteProgram(Work w)
      {
        if (w.TaskFinished)
        {
          Console.WriteLine("当前时间:{0}点 下班回家", w.Hour);
        }
      }
    }
  }

总结:

  这里用到的就是状态模式(State):当一个对象的内在状态改变时,允许改变其行为,这个对象看起来像是改变了其类。

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

  状态模式的好处是将与特定状态相关的行为局部化,并且将不同状态的行为分割开来。

注:文中所有代码及知识点均来自于《大话设计模式》,本人属于边学边看边敲代码边总结的阶段。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • C# 设计模式系列教程-状态模式

    1. 概述 当一个对象的内在状态改变时允许改变其行为,这个对象看起来像是改变了其类. 2. 解决的问题 主要解决的是当控制一个对象状态转换的条件表达式过于复杂时的情况.把状态的判断逻辑转移到表示不同的一系列类当中,可以把复杂的逻辑判断简单化. 3. 模式中的角色 3.1 上下文环境(Context):它定义了客户程序需要的接口并维护一个具体状态角色的实例,将与状态相关的操作委托给当前的Concrete State对象来处理. 3.2 抽象状态(State):定义一个接口以封装使用上下文环境的的一

  • C#学习笔记之状态模式详解

    本文通过例题为大家讲解C#学习笔记之状态模式,供大家参考,具体内容如下 题目1:通过代码描述每一天的不同时间段的工作效率 分析: 首先确定,工作状态指正在工作,或者以及下班这些情况,而这些情况所受影响的因素包括:当前时间以及任务是否已经完成.所以在Work中需要两个属性:hour和finish.然后根据这两个属性直接判断当前的工作状态即可. 实现: class Program { static void Main(string[] args) { //紧急项目 Work emergencyPro

  • Spring学习笔记1之IOC详解尽量使用注解以及java代码

    在实战中学习Spring,本系列的最终目的是完成一个实现用户注册登录功能的项目. 预想的基本流程如下: 1.用户网站注册,填写用户名.密码.email.手机号信息,后台存入数据库后返回ok.(学习IOC,mybatis,SpringMVC的基础知识,表单数据验证,文件上传等) 2.服务器异步发送邮件给注册用户.(学习消息队列) 3.用户登录.(学习缓存.Spring Security) 4.其他. 边学习边总结,不定时更新.项目环境为Intellij + Spring4. 一.准备工作. 1.m

  • java学习笔记之DBUtils工具包详解

    DBUtils工具包 一.介绍 DBUtils是Apache组织开源的数据库工具类. 二.使用步骤 ①.创建QueryRunner对象 ②.调用update()方法或者query()方法执行sql语句 三.构造方法及静态方法 QueryRunner类 1.构造方法 ①.无参构造 QueryRunner qr =new QueryRunner(); 使用无参构造的时候,调用update方法和query方法时就需要使用带Connection 类型参数的重载形式 ②.有参构造 QueryRunner

  • R语言学习笔记之lm函数详解

    在使用lm函数做一元线性回归时,发现lm(y~x+1)和lm(y~x)的结果是一致的,一直没找到两者之间的区别,经过大神们的讨论和测试,才发现其中的差别,测试如下: ------------------------------------------------------------- ------------------------------------------------------------- 结果可以发现,两者的结果是一样的,并无区别,但是若改为lm(y~x-1)就能看出+1和

  • go学习笔记读取consul配置文件详解

    目录 新建yaml文件 读取远程配置 新建yaml文件 在上文我们的 go学习笔记:使用 consul 做服务发现和配置共享 这里我们单独来用viper实现读取consul的配置, 我习惯与用yaml格式, 所以 首先 新建yaml文件 store: book: - author: john price: 10 - author: ken price: 12 bicycle: color: red price: 19.95 读取远程配置 可以直接调用viper.AddRemoteProvider

  • JavaScript设计模式之命令模式和状态模式详解

    目录 命令模式 命令模式介绍 代码实现 状态模式 状态模式介绍 代码实现 小结 命令模式 命令模式介绍 命令模式(Command)的定义是:用于将一个请求封装成一个对象,从而使你可用不同的请求对客户进行参数化:对请求排队或者记录请求日志,以及执行可撤销的操作. 也就是说改模式旨在将函数的调用.请求和操作封装成一个单一的对象,然后对这个对象进行一系列的处理.此外,可以通过调用实现具体函数的对象来解耦命令对象与接收对象. 代码实现 <!DOCTYPE html> <html lang=&qu

  • Go语言学习笔记之反射用法详解

    本文实例讲述了Go学习笔记之反射用法.分享给大家供大家参考,具体如下: 一.类型(Type) 反射(reflect)让我们能在运行期探知对象的类型信息和内存结构,这从一定程度上弥(mi)补了静态语言在动态行为上的不足.同时,反射还是实现元编程的重要手段. 和 C 数据结构一样,Go 对象头部并没有类型指针,通过其自身是无法在运行期获知任何类型相关信息的.反射操作所需要的全部信息都源自接口变量.接口变量除存储自身类型外,还会保存实际对象的类型数据. func TypeOf(i interface{

  • Symfony2学习笔记之模板用法详解

    本文实例讲述了Symfony2学习笔记之模板用法.分享给大家供大家参考,具体如下: 我们知道,controller负责处理每一个进入Symfony2应用程序的请求.实际上,controller把大部分的繁重工作都委托给了其它地方,以使代码能够被测试和重用.当一个controller需要生成HTML,CSS或者其他内容时,它把这些工作给了一个模板化引擎. 模板: 一个模板仅仅是一个文本文件,它能生成任意的文本格式(HTML,XML,CSV,LaTex...).最著名的模板类型就是PHP模板了,可以

  • 深入理解JavaScript系列(43):设计模式之状态模式详解

    介绍 状态模式(State)允许一个对象在其内部状态改变的时候改变它的行为,对象看起来似乎修改了它的类. 正文 举个例子,就比如我们平时在下载东西,通常就会有好几个状态,比如准备状态(ReadyState).下载状态(DownloadingState).暂停状态(DownloadPausedState).下载完毕状态(DownloadedState).失败状态(DownloadFailedState),也就是说在每个状态都只可以做当前状态才可以做的事情,而不能做其它状态能做的事儿. 由于Stat

  • Android编程设计模式之状态模式详解

    本文实例讲述了Android编程设计模式之状态模式.分享给大家供大家参考,具体如下: 一.介绍 状态模式中的行为是由状态来决定的,不同的状态下有不同的行为.状态模式和策略模式的结构几乎完全一样,但它们的目的.本质却完全不一样.状态模式的行为是平行的.不可替换的,策略模式的行为是彼此独立.可相互替换的.用一句话来表述,状态模式把对象的行为包装在不同的状态对象里,每一个状态对象都有一个共同的抽象状态基类.状态模式的意图是让一个对象在其内部状态改变的时候,其行为也随之改变. 二.定义 当一个对象的内在

随机推荐