学习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 相关文章推荐
Js 中debug方式
Feb 07 Javascript
对 lightbox JS 图片控件进行了一下改造, 使其他支持复杂的图片说明
Mar 20 Javascript
基于jQuery实现模拟页面加载进度条
Apr 01 Javascript
Bootstrap Metronic完全响应式管理模板学习笔记
Jul 08 Javascript
AngularJS ng-change 指令的详解及简单实例
Jul 30 Javascript
js实现表单及时验证功能 用户信息立即验证
Sep 13 Javascript
一个炫酷的Bootstrap导航菜单
Dec 28 Javascript
jquery操作select取值赋值与设置选中实例
Feb 28 Javascript
vue - vue.config.js中devServer配置方式
Oct 30 Javascript
微信小程序获取复选框全选反选选中的值(实例代码)
Dec 17 Javascript
详解JavaScript匿名函数和闭包
Jul 10 Javascript
利用PHP实现递归删除链表元素的方法示例
Oct 23 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
php ZipArchive压缩函数详解实例
2013/11/06 PHP
Convert Seconds To Hours
2007/06/16 Javascript
基于jquery跨浏览器显示的file上传控件
2011/10/24 Javascript
基于jquery的bankInput银行卡账号格式化
2012/08/22 Javascript
JS获取单击按钮单元格所在行的信息
2014/06/17 Javascript
JavaScript中的索引数组、关联数组和静态数组、动态数组讲解
2014/11/08 Javascript
深入理解vue.js双向绑定的实现原理
2016/12/05 Javascript
underscore之Chaining_动力节点Java学院整理
2017/07/10 Javascript
jackson解析json字符串,首字母大写会自动转为小写的方法
2017/12/22 Javascript
jQuery+datatables插件实现ajax加载数据与增删改查功能示例
2018/04/17 jQuery
jQuery实现的手动拖动控制进度条效果示例【测试可用】
2018/04/18 jQuery
微信小程序自定义prompt组件步骤详解
2018/06/12 Javascript
JavaScript函数式编程(Functional Programming)高阶函数(Higher order functions)用法分析
2019/05/22 Javascript
在Express中提供静态文件的实现方法
2019/10/17 Javascript
TypeScript之调用栈的实现
2019/12/31 Javascript
react 生命周期实例分析
2020/05/18 Javascript
vue v-for 点击当前行,获取当前行数据及event当前事件对象的操作
2020/09/10 Javascript
[40:29]2018DOTA2亚洲邀请赛 4.7总决赛 LGD vs Mineski 第一场
2018/04/10 DOTA
Python的Django框架中消息通知的计数器实现教程
2016/06/13 Python
使用python3.5仿微软记事本notepad
2016/06/15 Python
python读取图片并修改格式与大小的方法
2018/07/24 Python
关于python多重赋值的小问题
2019/04/17 Python
解决pycharm运行程序出现卡住scanning files to index索引的问题
2019/06/27 Python
解决python cv2.imread 读取中文路径的图片返回为None的问题
2020/06/02 Python
python 实现压缩和解压缩的示例
2020/09/22 Python
仓库管理专业个人自我评价范文
2013/11/11 职场文书
表彰先进集体通报
2014/01/12 职场文书
便利店的创业计划书
2014/01/15 职场文书
爱情检讨书大全
2014/01/21 职场文书
行政助理工作职责范本
2014/03/04 职场文书
产品委托授权书范本
2014/09/16 职场文书
2014年学生会个人工作总结
2014/11/07 职场文书
小学音乐教师个人工作总结
2015/02/05 职场文书
会计岗位工作总结
2015/08/12 职场文书
Java生成日期时间存入Mysql数据库的实现方法
2022/03/03 Java/Android
Apache Hudi数据布局黑科技降低一半查询时间
2022/03/31 Servers