状态(State)

意图

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

结构

状态模式结构图

适用性

  • 一个对象的行为取决于它的状态, 并且它必须在运行时刻根据状态改变它的行为;
  • 一个操作中含有庞大的多分支的条件语句,且这些分支依赖于该对象的状态。

效果

  • 把不同状态(State)的相关行为封装成一个个独立的对象,也便于以后增加新的状态和转换;
  • 代码结构更加清晰,状态(State)之间的转换意图更为明确;
  • 当状态(State)只是以编码类型做区分时,状态对象可以被共享。

思考

  • 状态之间的切换由谁执行?
    1. Context 负责调用当前状态的行为并更改当前状态;
    2. State负责履行自身的行为并更改Context当前状态。
  • 状态之间的转换关系也可以使用表查询(配置)方式实现;
  • 状态若存储有大量信息,频繁切换会产生大量的创建和销毁的开销;


示例

设计一个熔断器(CircuitBreaker)提供系统过载保护,防止应用程序不断地尝试执行可能会失败的操作。熔断器可以使用状态机来实现,内部模拟以下几种状态:

  1. 闭合(closed)状态
    对应用程序的请求能够直接引起方法的调用。代理类维护了最近调用失败的次数,如果某次调用失败,则使失败次数加1。如果最近失败次数超过了在给定时间内允许失败的阈值,则代理类切换到断开(Open)状态。此时代理开启了一个超时时钟,当该时钟超过了该时间,则切换到半断开(Half-Open)状态。该超时时间的设定是给了系统一次机会来修正导致调用失败的错误。
  • 断开(Open)状态
    在该状态下,对应用程序的请求会立即返回错误响应。
  • 半断开(Half-Open)状态
    允许对应用程序的一定数量的请求可以去调用服务。如果这些请求对服务的调用成功,那么可以认为之前导致调用失败的错误已经修正,此时熔断器切换到闭合状态(并且将错误计数器重置);如果这一定数量的请求有调用失败的情况,则认为导致之前调用失败的问题仍然存在,熔断器切回到断开方式,然后开始重置计时器来给系统一定的时间来修正错误。半断开状态能够有效防止正在恢复中的服务被突然而来的大量请求再次拖垮。

实现(C#)

熔断器的状态转换图,根据要求设计如下:

熔断器的状态转换图
using System;

public abstract class State
{
    protected readonly CircuitBreaker breaker;

    protected State (CircuitBreaker breaker)
    {
        this.breaker = breaker;
    }

    public abstract void Handle(Action handler);
}

public sealed class ClosedState : State
{
    private int failureCount = 0;
    private DateTime failureExpiredUtc;

    public ClosedState(CircuitBreaker breaker) : base(breaker)
    {
        this.failureExpiredUtc= DateTime.UtcNow.Add(breaker.FailureExpired);
    }

    public override void Handle(Action handler)
    {
        try
        {
            handler();
        }
        catch
        {
            // 超时,重新计数
            if(failureExpiredUtc < DateTime.UtcNow)
            {
                // reset
                this.failureExpiredUtc= DateTime.UtcNow.Add(breaker.FailureExpired);
                this.failureCount = 1;
            }
            else
            {
                this.failureCount ++;
            }
            
            //Console.WriteLine("failureCount:{0}", this.failureCount);
            // 在限定的时间内,错误次数达到一定阀值
            if(this.failureCount >= this.breaker.FailureCountThreshold && this.failureExpiredUtc > DateTime.UtcNow)
            {
                this.breaker.MoveToOpen();
            }
        }
    }

    public override string ToString()
    {
        return "闭合";
    }
}

public sealed class HalfOpenState : State
{
    private int successCount = 0;

    public HalfOpenState(CircuitBreaker breaker) : base(breaker) {}

    public override void Handle(Action handler)
    {
        try
        {
            handler();
            this.successCount ++;

            if(this.successCount >= this.breaker.SuccessCountThreshold)
            {
                this.breaker.MoveToClosed();
            }
        }
        catch
        {
            this.breaker.MoveToOpen();
        }
    }

    public override string ToString()
    {
        return "半开";
    }
}

public sealed class OpenState : State
{
    private System.Threading.Timer timer;

    public OpenState(CircuitBreaker breaker) : base(breaker)
    {
        this.timer = new System.Threading.Timer(this.MoveToHalfOpen,this, breaker.DelayTime, breaker.DelayTime);
    }

    public override void Handle(Action handler)
    {
        Console.WriteLine("无法获取远程资源.");
    }

    public void MoveToHalfOpen(object state)
    {
        this.breaker.MoveToHalfOpen();
    }

    public override string ToString()
    {
        return "断开";
    }
}

public sealed class CircuitBreaker
{
    public int FailureCountThreshold { get; private set; }
    public int SuccessCountThreshold { get; private set; }
    public TimeSpan DelayTime { get; private set; }
    public TimeSpan FailureExpired { get; private set; }
    public State State { get; private set; }

    public CircuitBreaker(int failureCountThreshold, int successCountThreshold, int delaySeconds, int failureSeconds)
    {
        this.FailureCountThreshold = failureCountThreshold;
        this.SuccessCountThreshold = successCountThreshold;
        this.DelayTime = TimeSpan.FromSeconds(delaySeconds);
        this.FailureExpired = TimeSpan.FromSeconds(failureSeconds);
        this.State = new ClosedState(this);
    }

    public void Execute(Action exec)
    {
        this.State.Handle(exec);
    }

    public void MoveToHalfOpen()
    {
        this.State = new HalfOpenState(this);
    }

    public void MoveToOpen()
    {
        this.State = new OpenState(this);
    }

    public void MoveToClosed()
    {
        this.State = new ClosedState(this);
    }

}

public class App
{
    public static void Main(string[] args)
    {
        // 1. 关闭: 在10秒内失败次数达到3次,状态调整为「断开」
        // 2. 断开: 15秒后,状态调整为「半开」
        // 3. 半开: 连续成功5次,状态调整为「闭合」
        CircuitBreaker breaker = new CircuitBreaker(3,5,15,10);

        breaker.Execute(() => Console.WriteLine("成功获得远程资源,当前状态:「{0}」", breaker.State));

        // 连续三次失败
        breaker.Execute(() => { throw new Exception(); });
        breaker.Execute(() => { throw new Exception(); });
        breaker.Execute(() => { throw new Exception(); });
        Console.WriteLine("连续3次失败后,当前状态:「{0}」", breaker.State);

        // 等待15秒        
        Console.Write("等待15秒,当前状态:「{0}」... ", breaker.State);
        System.Threading.Thread.Sleep(15500);
        Console.WriteLine("完毕! 当前状态:「{0}」", breaker.State);

        // 连续成功5次
        breaker.Execute(() => Console.WriteLine("1.成功获得远程资源,当前状态:「{0}」", breaker.State));
        breaker.Execute(() => Console.WriteLine("2.成功获得远程资源,当前状态:「{0}」", breaker.State));
        breaker.Execute(() => Console.WriteLine("3.成功获得远程资源,当前状态:「{0}」", breaker.State));
        breaker.Execute(() => Console.WriteLine("4.成功获得远程资源,当前状态:「{0}」", breaker.State));
        breaker.Execute(() => Console.WriteLine("5.成功获得远程资源,当前状态:「{0}」", breaker.State));
        Console.WriteLine("连续成功5次后,当前状态:「{0}」", breaker.State);

    }
}

// 控制台输出:
//  成功获得远程资源,当前状态:「闭合」
//  连续3次失败后,当前状态:「断开」
//  等待15秒,当前状态:「断开」... 完毕! 当前状态:「半开」
//  1.成功获得远程资源,当前状态:「半开」
//  2.成功获得远程资源,当前状态:「半开」
//  3.成功获得远程资源,当前状态:「半开」
//  4.成功获得远程资源,当前状态:「半开」
//  5.成功获得远程资源,当前状态:「半开」
//  连续成功5次后,当前状态:「闭合」
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335

推荐阅读更多精彩内容