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


Posted in Javascript onMarch 04, 2015

介绍

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

正文

举个例子,就比如我们平时在下载东西,通常就会有好几个状态,比如准备状态(ReadyState)、下载状态(DownloadingState)、暂停状态(DownloadPausedState)、下载完毕状态(DownloadedState)、失败状态(DownloadFailedState),也就是说在每个状态都只可以做当前状态才可以做的事情,而不能做其它状态能做的事儿。

由于State模式描述了下载(Download)如何在每一种状态下表现出不同的行为。这一模式的关键思想就是引入了一个叫做State的抽象类(或JS里的函数)来表示下载状态,State函数(作为原型)为每个状态的子类(继承函数)声明了一些公共接口。其每个继承函数实现与特定状态相关的行为,比如DownloadingState和DownloadedState分别实现了正在下载和下载完毕的行为。这些行为可以通过Download来来维护。

让我们来实现一把,首先定义作为其他基础函数的原型的State函数:

var State = function () {
};
State.prototype.download = function () {

    throw new Error("该方法必须被重载!");

};
State.prototype.pause = function () {

    throw new Error("该方法必须被重载!");

};
State.prototype.fail = function () {

    throw new Error("该方法必须被重载!");

};
State.prototype.finish = function () {

    throw new Error("该方法必须被重载!");

};

我们为State的原型定义了4个方法接口,分别对应着下载(download)、暂停(pause)、失败(fail)、结束(finish)以便子函数可以重写。

在编写子函数之前,我们先来编写一个ReadyState函数,以便可以将状态传递给第一个download状态:

var ReadyState = function (oDownload) {

    State.apply(this);

    this.oDownload = oDownload;

};
ReadyState.prototype = new State();
ReadyState.prototype.download = function () {

    this.oDownload.setState(this.oDownload.getDownloadingState());

    // Ready以后,可以开始下载,所以设置了Download函数里的状态获取方法

 console.log("Start Download!");

};
ReadyState.prototype.pause = function () {

    throw new Error("还没开始下载,不能暂停!");

};
ReadyState.prototype.fail = function () {

    throw new Error("文件还没开始下载,怎么能说失败呢!");

};
ReadyState.prototype.finish = function () {

    throw new Error("文件还没开始下载,当然也不能结束了!");

};

该函数接收了一个Download维护函数的实例作为参数,Download函数用于控制状态的改变和获取(类似于中央控制器,让外部调用),ReadyState重写了原型的download方法,以便开始进行下载。我们继续来看Download函数的主要功能:

var Download = function () {

    this.oState = new ReadyState(this);

};
Download.prototype.setState = function (oState) {

    this.oState = oState;

};
// 对外暴露的四个公共方法,以便外部调用
Download.prototype.download = function () {

    this.oState.download();

};
Download.prototype.pause = function () {

    this.oState.pause();

};
Download.prototype.fail = function () {

    this.oState.fail();

};
Download.prototype.finish = function () {

    this.oState.finish();

};
//获取各种状态,传入当前this对象

Download.prototype.getReadyState = function () {

    return new ReadyState(this);

};
Download.prototype.getDownloadingState = function () {

    return new DownloadingState(this);

};
Download.prototype.getDownloadPausedState = function () {

    return new DownloadPausedState(this);

};
Download.prototype.getDownloadedState = function () {

    return new DownloadedState(this);

};
Download.prototype.getDownloadedFailedState = function () {

    return new DownloadFailedState(this);

};

Download函数的原型提供了8个方法,4个是对用于下载状态的操作行为,另外4个是用于获取当前四个不同的状态,这4个方法都接收this作为参数,也就是将Download实例自身作为一个参数传递给处理该请求的状态对象(ReadyState 以及后面要实现的继承函数),这使得状态对象比必要的时候可以访问oDownlaod。

接下来,继续定义4个相关状态的函数:

var DownloadingState = function (oDownload) {

    State.apply(this);

    this.oDownload = oDownload;

};
DownloadingState.prototype = new State();
DownloadingState.prototype.download = function () {

    throw new Error("文件已经正在下载中了!");

};
DownloadingState.prototype.pause = function () { this.oDownload.setState(this.oDownload.getDownloadPausedState());

    console.log("暂停下载!");

};
DownloadingState.prototype.fail = function () { this.oDownload.setState(this.oDownload.getDownloadedFailedState());

    console.log("下载失败!");

};
DownloadingState.prototype.finish = function () {

    this.oDownload.setState(this.oDownload.getDownloadedState());

    console.log("下载完毕!");

};

DownloadingState的主要注意事项就是已经正在下载的文件,不能再次开始下载了,其它的状态都可以连续进行。

var DownloadPausedState = function (oDownload) {

    State.apply(this);

    this.oDownload = oDownload;

};
DownloadPausedState.prototype = new State();
DownloadPausedState.prototype.download = function () {

    this.oDownload.setState(this.oDownload.getDownloadingState());

    console.log("继续下载!");

};
DownloadPausedState.prototype.pause = function () {

    throw new Error("已经暂停了,咋还要暂停呢!");

};
DownloadPausedState.prototype.fail = function () { this.oDownload.setState(this.oDownload.getDownloadedFailedState());

    console.log("下载失败!");

};
DownloadPausedState.prototype.finish = function () {

    this.oDownload.setState(this.oDownload.getDownloadedState());

    console.log("下载完毕!");

};

DownloadPausedState函数里要注意的是,已经暂停的下载,不能再次暂停。
var DownloadedState = function (oDownload) {

    State.apply(this);

    this.oDownload = oDownload;

};
DownloadedState.prototype = new State();
DownloadedState.prototype.download = function () {

    this.oDownload.setState(this.oDownload.getDownloadingState());

    console.log("重新下载!");

};
DownloadedState.prototype.pause = function () {

    throw new Error("对下载完了,还暂停啥?");

};
DownloadedState.prototype.fail = function () {

    throw new Error("都下载成功了,咋会失败呢?");

};
DownloadedState.prototype.finish = function () {

    throw new Error("下载成功了,不能再为成功了吧!");

};

DownloadedState函数,同理成功下载以后,不能再设置finish了,只能设置重新下载状态。

var DownloadFailedState = function (oDownload) {

    State.apply(this);

    this.oDownload = oDownload;

};
DownloadFailedState.prototype = new State();
DownloadFailedState.prototype.download = function () {

    this.oDownload.setState(this.oDownload.getDownloadingState());

    console.log("尝试重新下载!");

};
DownloadFailedState.prototype.pause = function () {

    throw new Error("失败的下载,也不能暂停!");

};
DownloadFailedState.prototype.fail = function () {

    throw new Error("都失败了,咋还失败呢!");

};
DownloadFailedState.prototype.finish = function () {

    throw new Error("失败的下载,肯定也不会成功!");

};

同理,DownloadFailedState函数的失败状态,也不能再次失败,但可以和finished以后再次尝试重新下载。

调用测试代码,就非常简单了,我们在HTML里演示吧,首先是要了jquery,然后有3个按钮分别代表:开始下载、暂停、重新下载。(注意在Firefox里用firebug查看结果,因为用了 console.log方法)。

<html>

<head>

    <link type="text/css" rel="stylesheet" href="http://www.cnblogs.com/css/style.css" />

    <title>State Pattern</title>

    <script type="text/javascript" src="/jquery.js"></script>

    <script type="text/javascript" src="Download.js"></script>

    <script type="text/javascript" src="states/State.js"></script>

    <script type="text/javascript" src="states/DownloadFailedState.js"></script>

    <script type="text/javascript" src="states/DownloadPausedState.js"></script>

    <script type="text/javascript" src="states/DownloadedState.js"></script>

    <script type="text/javascript" src="states/DownloadingState.js"></script>

    <script type="text/javascript" src="states/ReadyState.js"></script>

</head>

<body>

    <input type="button" value="开始下载" id="download_button" />

    <input type="button" value="暂停" id="pause_button" />

    <input type="button" value="重新下载" id="resume_button" />

    <script type="text/javascript">

        var oDownload = new Download();

        $("#download_button").click(function () {

            oDownload.download();

        });
        $("#pause_button").click(function () {

            oDownload.pause();

        });
        $("#resume_button").click(function () {

            oDownload.download();

        });

    </script>

</body>

</html>

总结

状态模式的使用场景也特别明确,有如下两点:

1.一个对象的行为取决于它的状态,并且它必须在运行时刻根据状态改变它的行为。
2.一个操作中含有大量的分支语句,而且这些分支语句依赖于该对象的状态。状态通常为一个或多个枚举常量的表示。

Javascript 相关文章推荐
jQuery EasyUI API 中文文档 - ComboGrid 组合表格
Oct 13 Javascript
JavaScript 用Node.js写Shell脚本[译]
Sep 20 Javascript
javascript删除一个html元素节点的方法
Dec 20 Javascript
CKEditor无法验证的解决方案(js验证+jQuery Validate验证)
May 09 Javascript
jQuery simpleModal插件的使用介绍
Aug 30 Javascript
JavaScript实现大图轮播效果
Jan 11 Javascript
关于jquery form表单序列化的注意事项详解
Aug 01 jQuery
详解从Vue.js源码看异步更新DOM策略及nextTick
Oct 11 Javascript
vue watch深度监听对象实现数据联动效果
Aug 16 Javascript
vue中选项卡点击切换且能滑动切换功能的实现代码
Nov 25 Javascript
Vue实现搜索结果高亮显示关键字
May 28 Javascript
JavaScript实现移动端弹窗后禁止滚动
May 25 Javascript
深入理解JavaScript系列(42):设计模式之原型模式详解
Mar 04 #Javascript
javascript 动态创建表格的2种方法总结
Mar 04 #Javascript
深入理解JavaScript系列(41):设计模式之模板方法详解
Mar 04 #Javascript
深入理解JavaScript系列(40):设计模式之组合模式详解
Mar 04 #Javascript
百度地图自定义控件分享
Mar 04 #Javascript
jQuery实现仿淘宝带有指示条的图片转动切换效果完整实例
Mar 04 #Javascript
深入理解JavaScript系列(39):设计模式之适配器模式详解
Mar 04 #Javascript
You might like
PHP4(windows版本)中的COM函数
2006/10/09 PHP
php通过Chianz.com获取IP地址与地区的方法
2015/01/14 PHP
PHP递归调用数组值并用其执行指定函数的方法
2015/04/01 PHP
详细解读php的命名空间(二)
2018/02/21 PHP
php微信公众号开发之音乐信息
2018/10/20 PHP
如何在一个页面显示多个百度地图
2013/04/07 Javascript
jquery.autocomplete修改实现键盘上下键自动填充示例
2013/11/19 Javascript
禁止页面刷新让F5快捷键及右键都无效
2014/01/22 Javascript
Javascript单元测试框架QUnitjs详细介绍
2014/05/08 Javascript
JavaScript实现带箭头标识的多级下拉菜单效果
2015/08/27 Javascript
基于BootStrap环境写jQuery tabs插件
2016/07/12 Javascript
JavaScript 函数模式详解及示例
2016/09/07 Javascript
jQuery EasyUI 组件加上“清除”功能实例详解
2017/04/11 jQuery
基于JavaScript实现的快速排序算法分析
2017/04/14 Javascript
简单的JS控制button颜色随点击更改的实现方法
2017/04/17 Javascript
如何在js代码中消灭for循环实例详解
2018/07/29 Javascript
jQuery实现的响应鼠标移动方向插件用法示例【附源码下载】
2018/08/28 jQuery
Vue源码解析之Template转化为AST的实现方法
2018/12/14 Javascript
发布一款npm包帮助理解npm的使用
2019/01/03 Javascript
原生js实现each方法实例代码详解
2019/05/27 Javascript
vue 获取视频时长的实例代码
2019/08/20 Javascript
使用vscode快速建立vue模板过程详解
2019/10/10 Javascript
[08:08]DOTA2-DPC中国联赛2月28日Recap集锦
2021/03/11 DOTA
跟老齐学Python之编写类之四再论继承
2014/10/11 Python
Python Socket实现简单TCP Server/client功能示例
2017/08/05 Python
特征脸(Eigenface)理论基础之PCA主成分分析法
2018/03/13 Python
Python autoescape标签用法解析
2020/01/17 Python
python tkinter GUI绘制,以及点击更新显示图片代码
2020/03/14 Python
canvas探照灯效果的示例代码
2018/11/30 HTML / CSS
我看到了用指针调用函数的不同语法形式
2014/07/16 面试题
十岁生日家长答谢词
2014/01/17 职场文书
给护士表扬信
2014/01/19 职场文书
学校消防安全制度
2014/01/30 职场文书
测控技术自荐信
2014/06/05 职场文书
篮球兴趣小组活动总结
2014/07/07 职场文书
2015国庆节放假通知范文
2015/07/30 职场文书