【设计模式】使用 Builder 模式优化参数极多的函数

Builder(生成器)模式,让开发者能一步一步地创建复杂的对象。

优化参数极多的函数

常规方法定义的函数

一些函数,比如提交表单,需要的参数非常多,按照常规的写法,可能会定义为这样:

js 复制代码
function submit(name, phone, idCard, age, ...more) {}

这样定义的问题是:需要将参数写在固定的位置,随着参数的增多,这种传参方式会变得难以维护。

如果遇到某些定义在中间位置的参数为可选参数时,比如上面定义的 idCard,传参就变成了下面这样:

js 复制代码
submit('zkb', '188', '', 18, ...)

将参数合并到一个对象中传入

js 复制代码
function submit(params) {
    const name = params.name
    const phone = params.phone
    // ...
}

将参数合并到一个对象中传入可以缓解上面的问题,但这种方法也有缺点。

必须查看文档或者到方法的源码中才能知道这个方法具体需要什么参数。

使用 Builder 模式优化

Builder 模式改造的思想就是,不直接调用函数,而是提供一个生成器函数,在生成器函数中拆解函数的参数,分段存入,最后再执行 submit() 函数。

js 复制代码
function submit(params) {
    const name = params.name
    const phone = params.phone
    // ...
}

class SubmitBuilder() {
    name // 名称
    phone // 手机号
    setIdCard // 身份证号

    get age() {
        if(this.idCard) {
            // 从身份证号中提取
        }
        return ''
    }
    // ...
    setBaseInfo(name, phone) {
        this.name = name
        this.phone = phone
    }

    setIdCard(idCard) {
        this.idCard = idCard
    }

    invoke() {
        submit({
            name: this.name,
            phone: this.phone,
            // ...
        })
    }
}

这时提交表单时的操作就变成了:

js 复制代码
const submitBuilder = new SubmitBuilder()

submitBuilder
    .setBaseInfo('zkb', '188')
    .setIdCard('')
    .invoke()

可以看到,Builder 模式让代码变得比原来清晰许多,这样做有这些好处:

  • 让开发者不用直接去调用复杂的方法,而是可以再辅助类的引导下一步一步填充参数。
  • 如果参数之间有联动关系,可以在 set() 方法中去操作。
  • 如果一个参数可以由另一个参数推导出来,可以用 getter 来定义前者。
  • 可以确保某些参数必须同时传。
  • 添加参数时易于扩展、维护。

优化构造函数

可以用同样的方式优化构造函数,只是最后执行操作的方法名改为 build,当然这只是一个建议。

参考

《Node.js设计模式》

相关推荐
Moment7 小时前
Cursor 2.0 支持模型并发,我用国产 RWKV 模型实现了一模一样的效果 🤩🤩🤩
前端·后端·openai
狂炫冰美式8 小时前
QuizPort 1.0 · 让每篇好文都有测验陪跑
前端·后端·面试
安冬的码畜日常8 小时前
【JUnit实战3_27】第十六章:用 JUnit 测试 Spring 应用:通过实战案例深入理解 IoC 原理
spring·观察者模式·设计模式·单元测试·ioc·依赖注入·junit5
咋吃都不胖lyh8 小时前
.docx 和 .doc 是 Microsoft Word 文档的两种主要文件格式
前端·html·xhtml
哈乐8 小时前
网工应用题:配置命令补全类题目
服务器·前端·网络
uuai8 小时前
echarts不同版本显示不一致问题
前端·javascript·echarts
自然 醒8 小时前
企业微信自建应用开发详细教程,如何获取授权链接?如何使用js-sdk?
javascript·vue.js·企业微信
AKclown8 小时前
基于Monaco的diffEditor实现内容对比
前端·vue.js·react.js
Moonbit8 小时前
MoonBit Pearls Vol.12:初探 MoonBit 中的 Javascript 交互
javascript·后端·面试
chenbin___9 小时前
react native中 createAsyncThunk 的详细说明,及用法示例(转自通义千问)
javascript·react native·react.js