Iwen's blog Iwen's blog
首页
  • 前端文章

    • JavaScript
    • Vue
  • 学习笔记

    • 《JavaScript教程》笔记
    • 《JavaScript高级程序设计》笔记
    • 《ES6 教程》笔记
    • 《Vue》笔记
    • 《TypeScript 从零实现 axios》
    • 小程序笔记
  • HTML
  • CSS
  • 技术文档
  • GitHub技巧
  • Nodejs
  • 博客搭建
  • Linux
  • 学习
  • 面试
  • 心情杂货
  • 友情链接
  • 网站
  • 资源
  • Vue资源
  • 分类
  • 标签
  • 归档
复盘
关于

Iwen

不摸鱼的哥哥
首页
  • 前端文章

    • JavaScript
    • Vue
  • 学习笔记

    • 《JavaScript教程》笔记
    • 《JavaScript高级程序设计》笔记
    • 《ES6 教程》笔记
    • 《Vue》笔记
    • 《TypeScript 从零实现 axios》
    • 小程序笔记
  • HTML
  • CSS
  • 技术文档
  • GitHub技巧
  • Nodejs
  • 博客搭建
  • Linux
  • 学习
  • 面试
  • 心情杂货
  • 友情链接
  • 网站
  • 资源
  • Vue资源
  • 分类
  • 标签
  • 归档
复盘
关于
  • Vue

  • Vue进阶

  • CSS

  • ES6

  • Base

  • Core

  • Array

  • Object

  • String

  • Async

  • Browser

  • Http

  • 性能优化

  • 正则

  • 经典总结

  • 设计模式

    • 单例模式
    • 策略模式
    • 观察者模式
    • 代理模式
    • 职责链模式
  • 数据结构

  • 算法

  • 手写

  • TypeScript

  • 复盘
  • 设计模式
Mr.w
2020-11-29

策略模式

# 策略模式

定义一系列的算法,把它们一个个封装起来,并且使它们可以相互替换。

适用于业务场景中需要判断多种条件,甚至包含复杂条件嵌套的,可以使用策略模式来提升代码的可维护性和可读性。

比如支付,权限校验,各种复杂的判断场景。

/* 策略类 */
var levelOBJ = {
    "A": (money) => {
        return money * 4;
    },
    "B": (money) => {
        return money * 3;
    },
    "C": (money) => {
        return money * 2;
    } 
}

/* 环境类 */
var calculateBouns = (level, money) => {
    return levelOBJ[level](money);
}

calculateBouns('A', 10000) // 40000
calculateBouns('B', 20000) // 60000
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

实现一个表单校验:

<input type='text' name='username'>
<input type='password' name='passsword'>
<button>提交</button>
1
2
3

验证规则如下:

const button = document.querySelector('button')
const text = document.querySelector('input[type="text"]')
const password = document.querySelector('input[type="password"]')

button.onclick = () => {
    if(text.value === '') {
        console.log('用户名不能为空')
        return false;
    }
    if(password.value.length < 6) {
        console.log('密码不能少于6位')
        return false;
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14

这是一个很常见的思路

下面,使用策略模式重构表单校验,新增可扩展性的方法,可根据需求场景来自由发挥。

class Validator {
    constructor() {
        /**
         * 验证列表
         * 验证规则
         */
        this.rules = []
        this.strategies = {
            isEmpty: (val, msg) => {
                if (!val) return msg
            },
            minLength: (val, msg) => {
                if (val.length < 10) {
                    return msg
                }
            }
        }
    }
    // 添加验证内容
    add(val, rule, msg) {
        this.rules.push(() => {
            return this.strategies[rule].call(null, val, msg)
        })
    }
    // 新增验证规则
    addType(val, fn) {
        this.strategies[val] = fn
    }
    // 开始校验,返回校验后的结果
    star() {
        for (let i of this.rules) {
            const result = i()
            if (result) {
                return result
            }
        }
    }
}

const button = document.querySelector('button')
const text = document.querySelector('input[type="text"]')
const password = document.querySelector('input[type="password"]')

button.onclick = () => {
    const validator = new Validator()

    // validator.addType('mobile', (val, msg) => {
    //     if (!/(^1[3|5|8][0-9]{9}$)/.test(val)) {
    //         return msg
    //     }
    // })
    validator.add(text.value, 'isEmpty', '用户名不能为空')
    validator.add(password.value, 'minLength', '密码不能少于10位')

    const result = validator.star()
    if (result) {
        alert(result)
    } else {
        // ..success
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
单例模式
观察者模式

← 单例模式 观察者模式→

最近更新
01
flex布局页面自适应滚动条问题
12-28
02
前后端分离开发请求接口跨域如何携带cookie的问题
12-28
03
怎么实现div长宽比固定width-height4比3
12-28
更多文章>
Theme by Vdoing | Copyright © 2017-2022 Iwen | MIT License
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式