学习JavaScript设计模式之策略模式


Posted in Javascript onJanuary 12, 2016

把不变的部分和变化的部分隔开是每个设计模式的主题。

  • 条条大路通罗马。我们经常会遇到解决一件事情有多种方案,比如压缩文件,我们可以使用zip算法、也可以使用gzip算法。其灵活多样,我们可以采用策略模式解决。

一、定义

定义一系列的算法,把它们一个个封装起来,并且使它们可以相互替换。
基于策略类模式的程序至少由两部分组成。第一个部分是一组策略类,策略类封装了具体的算法,并负责具体的计算过程。第二个部分是环境类Context,Context接收客户的请求,随后把请求委托给某一个策略类。

二、示例

计算奖金。绩效为S的发放4倍工资,绩效为A的发放3倍工资,绩效为B的发放2倍工资。

var strategies = {
 "S": function(salary) {
  return salary * 4;
 },
 "A": function(salary) {
  return salary * 3;
 },
 "B": function(salary) {
  return salary * 2;
 }
};

// 接收请求
var calculateBonus = function(level, salary) {
 return strategies[level](salary);
};

// 测试
console.log(calculateBonus("S", 20000));
console.log(calculateBonus("A", 20000));
console.log(calculateBonus("B", 20000));

三、延伸:表单验证

/* 校验策略对象 */
var validateStrategies = {
 isEmpty: function (val, errorMsg) {
  if (!val) {
   return errorMsg;
  }
 },
 isURL: function (val, errorMsg) {
  if (!new RegExp("^(http:\\/\\/|https:\\/\\/)?([\\w\\-]+\\.)+[\\w\\-]+(\\/[\\w\\-\\.\\/?\\@\\%\\!\\&=\\+\\~\\:\\#\\;\\,]*)?$").test(val)) {
   return errorMsg;
  }
 },
 isEmail: function (val, errorMsg) {
  if (!new RegExp('\\w+((-\\w+)|(\\.\\w+))*\\@[A-Za-z0-9]+((\\.|-)[A-Za-z0-9]+)*\\.[A-Za-z0-9]+').test(val)) {
   return errorMsg;
  }
 },
 isMaxLength: function (val, length, errorMsg) {
  if (val.length > length) {
   return errorMsg;
  }
 },
 isMinLength: function (val, length, errorMsg) {
  if (val.length < length) {
   return errorMsg;
  }
 }
};

/* validator类 */
var validator = function () {
 // 缓存验证策略
 this.cache = [];
};

/**
 * 添加要验证的策略
 * @param dom  要验证的dom元素
 * @param rules  验证规则
 */
validator.prototype.add = function (dom, rules) {
 var self = this;
 for (var i = 0, rule; rule = rules[i++];) {
  (function (rule) {
   var strategyAry = rule.strategy.split(":");  // ["isMaxLength", "10"]
   var errorMsg = rule.errorMsg;     // "内容长度不能超过10"
   self.cache.push(function () {
    var strategy = strategyAry.shift();   // "isMaxLength"
    strategyAry.unshift(dom.value);    // ["1@qq", "10"]
    strategyAry.push(errorMsg);     // ["1@qq", "10", "内容长度不能超过10"]
    return validateStrategies[strategy].apply(dom, strategyAry);
   });
  })(rule);
 }
};

/* 开始校验 */
validator.prototype.start = function () {
 for (var i = 0, validateFunc; validateFunc = this.cache[i++];) {
  var errorMsg = validateFunc();
  if (errorMsg) {
   return errorMsg;
  }
 }
};


// 测试

<label for="email">邮箱:</label><input type="text" name="email" value="1@qq">

var validatorInstance = new validator();
validatorInstance.add(
 document.getElementsByName("email")[0], 
 [{
  strategy: "isEmpty",
  errorMsg: "内容不能为空"
 },{
  strategy: "isMaxLength:10",
  errorMsg: "内容长度不能超过10"
 },{
  strategy: "isEmail",
  errorMsg: "email格式不对"
 }]);
errorMsg = validatorInstance.start();

希望本文所述对大家学习javascript程序设计有所帮助。

Javascript 相关文章推荐
jQuery EasyUI 开源插件套装 完全替代ExtJS
Mar 24 Javascript
js各种验证文本框输入格式(正则表达式)
Oct 22 Javascript
js清空表单数据的两种方式(遍历+reset)
Jul 18 Javascript
谷歌浏览器调试JavaScript小技巧
Dec 29 Javascript
浅谈轻量级js模板引擎simplite
Feb 13 Javascript
js+html5实现canvas绘制镂空字体文本的方法
Jun 05 Javascript
JavaScript实现自动消除按钮功能的方法
Aug 05 Javascript
jQuery的Ajax用户认证和注册技术实例教程(附demo源码)
Dec 08 Javascript
JavaScript深度复制(deep clone)的实现方法
Feb 19 Javascript
浅谈原生JS中的延迟脚本和异步脚本
Jul 12 Javascript
在vue项目中引入vue-beauty操作方法
Feb 11 Javascript
vue3.0 上手体验
Sep 21 Javascript
基于jQuery1.9版本如何判断浏览器版本类型
Jan 12 #Javascript
jQuery版本升级踩坑大全
Jan 12 #Javascript
基于jQuery实现点击最后一行实现行自增效果的表格
Jan 12 #Javascript
7个jQuery最佳实践
Jan 12 #Javascript
实例详解jQuery Mockjax 插件模拟 Ajax 请求
Jan 12 #Javascript
JavaScript实现输入框(密码框)出现提示语
Jan 12 #Javascript
javascript自动恢复文本框点击清除后的默认文本
Jan 12 #Javascript
You might like
咖啡的化学
2021/03/03 咖啡文化
MySQL相关说明
2007/01/15 PHP
Joomla下利用configuration.php存储简单数据
2010/05/19 PHP
PHP面向对象之后期静态绑定功能介绍
2015/05/18 PHP
详解Yii2.0 rules验证规则集合
2017/03/21 PHP
php5.6.x到php7.0.x特性小结
2019/08/17 PHP
PHP使用观察者模式处理异常信息的方法详解
2019/09/24 PHP
JavaScript Function函数类型介绍
2015/04/08 Javascript
jquery专业的导航菜单特效代码分享
2015/08/29 Javascript
jquery实现美观的导航菜单鼠标提示特效代码
2015/09/06 Javascript
基于jQuery实现文本框只能输入数字(小数、整数)
2016/01/14 Javascript
微信小程序开发之选项卡(窗口底部TabBar)页面切换
2017/04/12 Javascript
Ionic + Angular.js实现图片轮播的方法示例
2017/05/21 Javascript
vue 如何添加全局函数或全局变量以及单页面的title设置总结
2017/06/01 Javascript
vue组件文档(.md)中如何自动导入示例(.vue)详解
2019/01/25 Javascript
详解Vue中组件的缓存
2019/04/20 Javascript
Vue中实现权限控制的方法示例
2019/06/07 Javascript
解析原来浏览器原生支持JS Base64编码解码
2019/08/12 Javascript
Element 默认勾选表格 toggleRowSelection的实现
2019/09/04 Javascript
[05:53]完美世界携手游戏风云打造 卡尔工作室观战系统篇
2013/04/22 DOTA
python编写的最短路径算法
2015/03/25 Python
Python中使用gzip模块压缩文件的简单教程
2015/04/08 Python
linux下python抓屏实现方法
2015/05/22 Python
Python 实现淘宝秒杀的示例代码
2018/01/02 Python
python实现矩阵打印
2019/03/02 Python
python 对字典按照value进行排序的方法
2019/05/09 Python
利用pyuic5将ui文件转换为py文件的方法
2019/06/19 Python
利用matplotlib实现根据实时数据动态更新图形
2019/12/13 Python
pandas处理csv文件的方法步骤
2020/10/16 Python
Alpine安装Python3依赖出现的问题及解决方法
2020/12/25 Python
美国购买隐形眼镜网站:Lenses For Less
2020/07/05 全球购物
生产部厂长助理职位说明书
2014/03/03 职场文书
海洋科学专业求职信
2014/08/10 职场文书
2014年作风建设心得体会
2014/10/22 职场文书
幼儿园家长工作总结2015
2015/04/25 职场文书
Java日常练习题,每天进步一点点(38)
2021/07/26 Java/Android