学习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.ajax)
Nov 19 Javascript
Javascript实现简单二级下拉菜单实例
Jun 15 Javascript
Jquery api 速查表分享
Jan 12 Javascript
javascript学习笔记之函数定义
Jun 25 Javascript
Fullpage.js固定导航栏-实现定位导航栏
Mar 17 Javascript
jQuery3.0中的buildFragment私有函数详解
Aug 16 Javascript
基于vue2.0+vuex+localStorage开发的本地记事本示例
Feb 28 Javascript
微信JS-SDK实现微信会员卡功能(给用户微信卡包里发送会员卡)
Jul 25 Javascript
js定义类的方法示例【ES5与ES6】
Jul 30 Javascript
vue 动态表单开发方法案例详解
Dec 02 Javascript
JS访问对象两种方式区别解析
Aug 29 Javascript
JS前端使用canvas实现物体的点选示例
Aug 05 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
十大催泪虐心动漫,你能坚持看到第几部?
2020/03/04 日漫
全国FM电台频率大全 - 6 辽宁省
2020/03/11 无线电
PHP实现的英文名字全拼随机排号脚本
2014/07/04 PHP
本地计算机无法启动Apache故障处理
2014/08/08 PHP
常用PHP数组排序函数归纳
2016/08/08 PHP
PHP连接MySQL数据库并以json格式输出
2018/05/21 PHP
详解php中curl返回false的解决办法
2019/03/18 PHP
php多进程并发编程防止出现僵尸进程的方法分析
2020/02/28 PHP
使用git迁移Laravel项目至新开发环境的步骤详解
2020/04/06 PHP
javascript对象的使用和属性操作示例详解
2014/03/02 Javascript
jquery实现显示已选用户
2014/07/21 Javascript
js的flv视频播放器插件使用方法
2015/06/23 Javascript
JSP基于Bootstrap分页显示实例解析
2016/06/12 Javascript
JavaScript中的Reflect对象详解(ES6新特性)
2016/07/22 Javascript
javascript数组定义的几种方法
2017/10/06 Javascript
node中的session的具体使用
2018/09/14 Javascript
vue实现跨域的方法分析
2019/05/21 Javascript
Node.js 的 GC 机制详解
2019/06/03 Javascript
Vue解决移动端弹窗滚动穿透问题
2020/12/15 Vue.js
Python 经典面试题 21 道【不可错过】
2018/09/21 Python
python mac下安装虚拟环境的图文教程
2019/04/12 Python
PIL图像处理模块paste方法简单使用详解
2019/07/17 Python
学习和使用python的13个理由
2019/07/30 Python
Python 中list ,set,dict的大规模查找效率对比详解
2019/10/11 Python
Python命令行click参数用法解析
2019/12/19 Python
python+opencv3生成一个自定义纯色图教程
2020/02/19 Python
matlab中二维插值函数interp2的使用详解
2020/04/22 Python
浅谈Python中的字符串
2020/06/10 Python
Python制作一个仿QQ办公版的图形登录界面
2020/09/22 Python
美国半成品食材配送服务商:Home Chef
2018/01/25 全球购物
在校硕士自我鉴定
2014/01/23 职场文书
模特职业生涯规划范文
2014/02/26 职场文书
全国法制宣传日活动总结2014
2014/11/01 职场文书
横店影视城导游词
2015/02/06 职场文书
详解OpenCV曝光融合
2022/04/29 Python
Java 多线程并发FutureTask
2022/06/28 Java/Android