策略模式实现 Vue 动态表单验证的方法


Posted in Javascript onSeptember 16, 2019

策略模式(Strategy Pattern)又称政策模式,其定义一系列的算法,把它们一个个封装起来,并且使它们可以互相替换。封装的策略算法一般是独立的,策略模式根据输入来调整采用哪个算法。关键是策略的 实现和使用分离

注意:本文可能用到一些编码技巧比如 IIFE(Immediately Invoked Function Expression, 立即调用函数表达式),ES6 的语法 let/const、箭头函数、rest 参数,短路运算符 等,如果还没接触过可以点击链接稍加学习 ~

1. 你曾见过的策略模式

现在电子产品种类繁多,尺寸多种多样,有时候你会忍不住想拆开看看里面啥样(想想小时候拆的玩具车还有遥控器),但是螺丝规格很多,螺丝刀尺寸也不少,如果每碰到一种规格就买一个螺丝刀,家里就得堆满螺丝刀了。所以现在人们都用多功能的螺丝刀套装,螺丝刀把只需要一个,碰到不同规格的螺丝只要换螺丝刀头就行了,很方便,体积也变小很多。

策略模式实现 Vue 动态表单验证的方法

再举个栗子,一辆车的轮胎有很多规格,在泥泞路段开的多的时候可以用泥地胎,在雪地开得多可以用雪地胎,高速公路上开的多的时候使用高性能轮胎,针对不同使用场景更换不同的轮胎即可,不需更换整个车。

这些都是策略模式的实例,螺丝刀/车属于封装上下文,封装和使用不同的螺丝刀头/轮胎,螺丝刀头/轮胎这里就相当于策略,可以根据需求不同来更换不同的使用策略。

在这些场景中,有以下特点:

螺丝刀头/轮胎(策略)之间相互独立,但又可以相互替换;

螺丝刀/车(封装上下文)可以根据需要的不同选用不同的策略;

2. 实例的代码实现

具体的例子我们用编程上的例子来演示,比较好量化。

场景是这样的,某个电商网站希望举办一个活动,通过打折促销来销售库存物品,有的商品满 100 减 30,有的商品满 200 减 80,有的商品直接8折出售(想起被双十一支配的恐惧),这样的逻辑交给我们,我们要怎样去实现呢。

通过判断输入的折扣类型来计算计算商品总价的方式,几个 if-else 就满足了需求,但是这样的做法的缺点也很明显:

priceCalculate 函数随着折扣类型的增多,  if-else 判断语句会变得越来越臃肿;

如果增加了新的折扣类型或者折扣类型的算法有所改变,那么需要更改  priceCalculate 函数的实现,这是违反开放-封闭原则的;

可复用性差,如果在其他的地方也有类似这样的算法,但规则不一样,上述代码不能复用;

我们可以改造一下,将计算折扣的 算法部分提取出来 保存为一个对象,折扣的 类型作为 key ,这样索引的时候 通过对象的键值索引调用具体的算法

这样 算法的实现和算法的使用就被分开了 ,想添加新的算法也变得十分简单:

如果你希望计算算法隐藏起来,那么可以借助 IIFE 使用闭包的方式,这时需要添加增加策略的入口,以方便扩展:

这样算法就被隐藏起来,并且预留了增加策略的入口,便于扩展。

3. 策略模式的通用实现

根据上面的例子提炼一下策略模式,折扣计算方式可以被认为是策略(Strategy),这些策略之间可以相互替代,而具体折扣的计算过程可以被认为是封装上下文(Context),封装上下文可以根据需要选择不同的策略。

主要有下面几个概念:

Context:封装上下文,根据需要调用需要的策略,屏蔽外界对策略的直接调用,只对外提供一个接口,根据需要调用对应的策略;

Strategy:策略,含有具体的算法,其方法的外观相同,因此可以互相代替;

StrategyMap:所有策略的合集,供封装上下文调用;

结构图如下:

策略模式实现 Vue 动态表单验证的方法

下面使用通用化的方法实现一下。

通用实现看起来似乎比较简单,这里分享一下项目实战。

4. 实战中的策略模式

 4.1 表格 formatter

这里举一个 Vue + ElementUI 项目中用到的例子,其他框架的项目原理也类似,和大家分享一下。

Element 的表格控件的 Column 接受一个 formatter 参数,用来格式化内容,其类型为函数,并且还可以接受几个特定参数,像这样: Function(row,column,cellValue,index)

以文件大小转化为例,后端经常会直接传 bit 单位的文件大小,那么前端需要根据后端的数据,根据需求转化为自己需要的单位的文件大小,比如 KB/MB。

首先实现文件计算的算法:

那么在组件中我们可以直接:

代码实例可以参看 codepen - 策略模式实战

运行结果如下图:

策略模式实现 Vue 动态表单验证的方法

4.2 表单验证

除了表格中的 formatter 之外,策略模式也经常用在表单验证的场景,这里举一个 Vue + ElementUI 项目的例子,其他框架同理。

ElementUI 的 Form 表单 具有表单验证功能,用来校验用户输入的表单内容。实际需求中表单验证项一般会比较复杂,所以需要给每个表单项增加 validator 自定义校验方法。

我们可以像官网示例一样把表单验证都写在组件的状态 data 函数中,但是这样就不好复用使用频率比较高的表单验证方法了,这时我们可以结合策略模式和函数柯里化的知识来重构一下。首先我们在项目的工具模块(一般是 utils 文件夹)实现通用的表单验证方法:

然后在 utils/index.js 中增加一个柯里化方法,用来生成表单验证函数:

上面的 formValidateGene 函数接受两个参数,第一个是验证规则,也就是 src/utils/validates.js 文件中提取出来的通用验证规则的方法名,第二个参数是报错的话表单验证的提示信息。

可以看见在使用的时候非常方便,把表单验证方法提取出来作为策略,使用柯里化方法动态选择表单验证方法,从而对策略灵活运用,大大加快开发效率。

代码实例可以参看 codesandbox - 策略模式表单验证实战

运行结果:

策略模式实现 Vue 动态表单验证的方法

5. 策略模式的优缺点

策略模式将算法的 实现和使用拆分 ,这个特点带来了很多优点:

策略之间相互独立,但 策略可以自由切换 ,这个策略模式的特点给策略模式带来很多灵活性,也提高了策略的复用率;

如果不采用策略模式,那么在选策略时一般会采用多重的条件判断,采用策略模式可以 避免多重条件判断 ,增加可维护性;

可扩展性好,策略可以很方便的进行扩展;

策略模式的缺点:

策略相互独立,因此一些复杂的算法逻辑 无法共享 ,造成一些资源浪费;

如果用户想采用什么策略,必须了解策略的实现,因此 所有策略都需向外暴露 ,这是违背迪米特法则/最少知识原则的,也增加了用户对策略对象的使用成本。

6. 策略模式的适用场景

那么应该在什么场景下使用策略模式呢:

多个算法 只在行为上稍有不同 的场景,这时可以使用策略模式来动态选择算法;

算法 需要自由切换 的场景;

有时 需要多重条件判断 ,那么可以使用策略模式来规避多重条件判断的情况;

7. 其他相关模式

7.1 策略模式和模板方法模式

策略模式和模板方法模式的作用比较类似,但是结构和实现方式有点不一样。

策略模式让我们在程序运行的时候动态地指定要使用的算法;

模板方法模式是在子类定义的时候就已经确定了使用的算法;

7.2 策略模式和享元模式

总结

 以上所述是小编给大家介绍的策略模式实现 Vue 动态表单验证的方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

Javascript 相关文章推荐
Web层改进II-用xmlhttp 无声息提交复杂表单
Jan 22 Javascript
一个简单的JavaScript数据缓存系统实现代码
Oct 24 Javascript
你需要知道的10个最佳javascript开发实践小结
Apr 15 Javascript
JS 获取浏览器和屏幕宽高等信息的实现思路及代码
Jul 31 Javascript
js判断字符长度以及中英文数字等
Dec 31 Javascript
jQuery leonaScroll 1.1 自定义滚动条插件(推荐)
Sep 17 Javascript
easyUI下拉列表点击事件使用方法
May 18 Javascript
vue中Axios的封装与API接口的管理详解
Aug 09 Javascript
详解React 元素渲染
Jul 07 Javascript
Vue 解决通过this.$refs来获取DOM或者组件报错问题
Jul 28 Javascript
JS typeof fn === 'function' && fn()详解
Aug 22 Javascript
如何在vue中使用video.js播放m3u8格式的视频
Feb 01 Vue.js
jQuery设置下拉框显示与隐藏效果的方法分析
Sep 15 #jQuery
Vue实现滑动拼图验证码功能
Sep 15 #Javascript
Vue 利用指令实现禁止反复发送请求的两种方法
Sep 15 #Javascript
解决layui调用自定义方法提示未定义的问题
Sep 14 #Javascript
layui使用label标签的方法
Sep 14 #Javascript
使用layui定义一个模块并使用的例子
Sep 14 #Javascript
基于Layui自定义模块的使用方法详解
Sep 14 #Javascript
You might like
phpStudy配置多站点多域名和多端口的方法
2017/09/01 PHP
用Laravel轻松处理千万级数据的方法实现
2020/12/25 PHP
url 编码 js url传参中文乱码解决方案
2010/04/11 Javascript
js实现连续英文字符自动换行兼容ie6 ie7和firefox
2013/09/06 Javascript
js 将图片连接转换成base64格式的简单实例
2016/08/10 Javascript
使用BootStrap实现表格隔行变色及hover变色并在需要时出现滚动条
2017/01/04 Javascript
微信小程序 template模板详解及实例
2017/02/21 Javascript
Vue中组件之间数据的传递的示例代码
2017/09/08 Javascript
JavaScript的setter与getter方法
2017/11/29 Javascript
详解新手使用vue-router传参时注意事项
2019/06/06 Javascript
react koa rematch 如何打造一套服务端渲染架子
2019/06/26 Javascript
在vue中使用axios实现post方式获取二进制流下载文件(实例代码)
2019/12/16 Javascript
详解JavaScript 事件流
2020/09/02 Javascript
Vue实现Header渐隐渐现效果的实例代码
2020/11/05 Javascript
[56:48]FNATIC vs EG 2019国际邀请赛小组赛 BO2 第二场 8.15
2019/08/16 DOTA
[01:02:30]Mineski vs Secret 2019国际邀请赛淘汰赛 败者组 BO3 第三场 8.22
2019/09/05 DOTA
python利用sklearn包编写决策树源代码
2017/12/21 Python
python机器学习案例教程——K最近邻算法的实现
2017/12/28 Python
利用python求积分的实例
2019/07/03 Python
Python 类属性与实例属性,类对象与实例对象用法分析
2019/09/20 Python
Python+OpenCV图像处理——打印图片属性、设置存储路径、调用摄像头
2020/10/22 Python
.net开发工程师面试题
2014/02/25 面试题
优秀教师的感人事迹
2014/02/04 职场文书
趣味运动会活动方案
2014/02/12 职场文书
亮化工程实施方案
2014/03/17 职场文书
初三开学计划书
2014/04/27 职场文书
电话客服工作职责
2014/07/27 职场文书
租房协议书范例
2014/10/14 职场文书
2014年质量工作总结
2014/11/22 职场文书
2014年教研员工作总结
2014/12/23 职场文书
小学生通知书评语
2014/12/31 职场文书
2015年收银工作总结范文
2015/04/01 职场文书
同学聚会通知书
2015/04/20 职场文书
该怎么书写道歉信?
2019/07/03 职场文书
导游词之湖州-太湖
2019/10/11 职场文书
HTML基础详解(上)
2021/10/16 HTML / CSS