策略模式实现 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 相关文章推荐
无缝滚动改进版支持上下左右滚动(封装成函数)
Dec 04 Javascript
jquery ajax实现下拉框三级无刷新联动,且保存保持选中值状态
Oct 29 Javascript
快速解决FusionCharts联动的中文乱码问题
Dec 04 Javascript
node.js中的fs.utimesSync方法使用说明
Dec 15 Javascript
简单讲解jQuery中的子元素过滤选择器
Apr 18 Javascript
深入理解javascript中concat方法
Dec 12 Javascript
超全面的javascript中变量命名规则
Feb 09 Javascript
vue-resourse将json数据输出实例
Mar 08 Javascript
jQuery+CSS3实现点赞功能
Mar 13 Javascript
js禁止表单重复提交
Aug 29 Javascript
JS中比较两个Object数组是否相等方法实例
Nov 11 Javascript
vue缓存之keep-alive的理解和应用详解
Nov 02 Javascript
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
提问的智慧(2)
2006/10/09 PHP
PHP设计模式之注册树模式分析
2018/01/26 PHP
Three.js源码阅读笔记(光照部分)
2012/12/27 Javascript
可兼容IE的获取及设置cookie的jquery.cookie函数方法
2013/09/02 Javascript
javascript实现根据汉字获取简拼
2016/09/25 Javascript
微信小程序 详解Page中data数据操作和函数调用
2017/01/12 Javascript
axios基本入门用法教程
2017/03/25 Javascript
详解Vue.js组件可复用性的混合(mixin)方式和自定义指令
2017/09/06 Javascript
浅谈vue项目重构技术要点和总结
2018/01/23 Javascript
React Native使用fetch实现图片上传的示例代码
2018/03/07 Javascript
React 组件间的通信示例
2018/06/14 Javascript
webpack手动配置React开发环境的步骤
2018/07/02 Javascript
vue实现记事本功能
2019/06/26 Javascript
react koa rematch 如何打造一套服务端渲染架子
2019/06/26 Javascript
解决layer.msg 不居中 ifram中的问题
2019/09/05 Javascript
layui table 获取分页 limit的方法
2019/09/20 Javascript
JavaScript定时器使用方法详解
2020/03/26 Javascript
js实现特别简单的钟表效果
2020/09/14 Javascript
Python中还原JavaScript的escape函数编码后字符串的方法
2014/08/22 Python
Python判断文件和字符串编码类型的实例
2017/12/21 Python
python opencv捕获摄像头并显示内容的实现
2019/07/11 Python
Python元组 tuple的概念与基本操作详解【定义、创建、访问、计数、推导式等】
2019/10/30 Python
Python函数默认参数常见问题及解决方案
2020/03/26 Python
Windows 下更改 jupyterlab 默认启动位置的教程详解
2020/05/18 Python
Pytorch 高效使用GPU的操作
2020/06/27 Python
Python的scikit-image模块实例讲解
2020/12/30 Python
浅谈html5 video 移动端填坑记
2018/01/15 HTML / CSS
入党转预备思想汇报
2014/01/07 职场文书
总经理司机职责
2014/02/02 职场文书
小学生国庆节演讲稿
2014/09/05 职场文书
2016年“七一建党节”广播稿
2015/12/18 职场文书
go结构体嵌套的切片数组操作
2021/04/28 Golang
Python OpenCV 图像平移的实现示例
2021/06/04 Python
带你了解CSS基础知识,样式
2021/07/21 HTML / CSS
MySQL中varchar和char类型的区别
2021/11/17 MySQL
浅谈Vue的computed计算属性
2022/03/21 Vue.js