动态参数下函数调用的策略模式实践

动态参数下函数调用的策略模式实践

本文探讨了在JavaScript/TypeScript中,如何优雅地处理根据不同业务场景(如面试类型)调用具有不同参数签名的函数。通过引入策略模式,我们将具体业务逻辑封装到独立的策略类中,从而实现核心调用逻辑的统一与灵活性,有效解决了在动态选择函数时参数不匹配的问题,提升了代码的可维护性和可扩展性。

挑战:动态函数调用与可变参数

在开发过程中,我们经常会遇到需要根据特定条件动态调用不同函数的场景。一个常见的例子是,根据面试类型(如技术面试或hr面试)来验证招聘人员。然而,这些验证函数可能需要不同的参数。

考虑以下初始设计:

import { getHrRecruiters, getRecruiters } from '../queue';import { validateTechnicalInterview } from './validateTechnicalInterview';import { matchHrRecruiters } from './matchHrRecruiters';import { THrInterviewer, THrRecruit, TRecruit } from '../../types';export const recruitersCategoryHandlers = {  TECHNICAL_INTERVIEW: {    getter: getRecruiters,    setter: {      validateRecruiters: (        recruiters: THrInterviewer[],        recruit: TRecruit | THrRecruit      ) => validateTechnicalInterview(recruiters, recruit),    },  },  HR_INTERVIEW: {    getter: getHrRecruiters,    setter: {      validateRecruiters: (        recruiters: THrInterviewer[],        recruit: TRecruit | THrRecruit,        param3: any,        param4: any      ) => matchHrRecruiters(recruiters, recruit as THrRecruit, param3, param4),    },  },};

当尝试统一调用 validateRecruiters 时,问题便浮现了:

const matchedSlots = getMatchingSlots(  recruitersCategoryHandlers[    interviewCategory as EInterviewCategory  ].setter.validateRecruiters(???), // 如何统一传递参数?  slotsWithEmail);

TECHNICAL_INTERVIEW 类型的 validateRecruiters 需要两个参数,而 HR_INTERVIEW 类型则需要四个参数。直接在调用点统一传递参数会变得非常困难,因为不同分支需要的参数数量和类型不一致,导致代码难以维护和扩展。

解决方案:策略模式 (Strategy Pattern)

为了解决上述问题,我们可以采用策略模式。策略模式允许在运行时选择算法的行为。它通过定义一系列算法,将每一个算法封装起来,并使它们可以相互替换。这意味着,我们可以为每种面试类型定义一个“策略”,每个策略都封装了其特有的 validateRecruiters 逻辑,同时提供一个统一的接口。

1. 定义策略接口

首先,我们需要定义一个公共的接口或抽象类,所有具体策略都将实现或继承它。这个接口将包含 validateRecruiters 方法,并使用 …params: any[] 来处理可变数量的参数,以确保接口的通用性。

// 定义策略接口interface ValidateRecruitersStrategy {  validateRecruiters(recruiters: THrInterviewer[], recruit: TRecruit | THrRecruit, ...params: any[]): any;}

2. 实现具体策略

接下来,为每种面试类型创建具体的策略类,实现 ValidateRecruitersStrategy 接口。每个策略类将封装其特定的 validateRecruiters 逻辑。

// 实现技术面试策略class TechnicalInterviewStrategy implements ValidateRecruitersStrategy {  validateRecruiters(recruiters: THrInterviewer[], recruit: TRecruit | THrRecruit): any {    // 技术面试只关心前两个参数    return validateTechnicalInterview(recruiters, recruit);  }}// 实现HR面试策略class HrInterviewStrategy implements ValidateRecruitersStrategy {  validateRecruiters(recruiters: THrInterviewer[], recruit: TRecruit | THrRecruit, ...params: any[]): any {    // HR面试需要额外的参数,通过解构或直接传递...params来获取    const [param3, param4] = params;    return matchHrRecruiters(recruiters, recruit as THrRecruit, param3, param4);  }}

注意:在 HrInterviewStrategy 中,我们通过 …params 接收所有额外参数,并根据需要解构或使用它们。而 TechnicalInterviewStrategy 虽然也接收 …params,但它只使用了前两个必需参数,忽略了其余的。这是策略模式处理可变参数的关键。

3. 重构处理器对象

现在,我们可以更新 recruitersCategoryHandlers 对象,使其不再直接包含匿名函数,而是引用具体的策略实例。

export const recruitersCategoryHandlers = {  TECHNICAL_INTERVIEW: {    getter: getRecruiters,    setter: new TechnicalInterviewStrategy(), // 使用策略实例  },  HR_INTERVIEW: {    getter: getHrRecruiters,    setter: new HrInterviewStrategy(), // 使用策略实例  },};

4. 统一调用策略方法

通过策略模式,现在可以统一地调用 validateRecruiters 方法,无论具体是哪种面试类型。所有可能的参数都可以在调用点传递,由具体的策略类决定如何使用它们。

// 假设已确定面试类别及其他所需参数const interviewCategory = 'HR_INTERVIEW'; // 示例:可以是动态获取的值const param3 = 'someValue3'; // 示例:根据业务逻辑传入const param4 = 'someValue4'; // 示例:根据业务逻辑传入const slotsWithEmail = {}; // 替换为实际值const initialRecruiters = recruitersCategoryHandlers[interviewCategory].getter(); // 获取初始招聘者// 统一调用 validateRecruitersconst matchedSlots = getMatchingSlots(  recruitersCategoryHandlers[interviewCategory].setter.validateRecruiters(    initialRecruiters,    slotsWithEmail, // 对应 recruit 参数    param3,         // 对应 param3    param4          // 对应 param4  ),  slotsWithEmail);

完整示例代码

import { getHrRecruiters, getRecruiters } from '../queue';import { validateTechnicalInterview } from './validateTechnicalInterview';import { matchHrRecruiters } from './matchHrRecruiters';import { THrInterviewer, THrRecruit, TRecruit } from '../../types'; // 假设类型定义存在// 1. 定义策略接口interface ValidateRecruitersStrategy {  validateRecruiters(recruiters: THrInterviewer[], recruit: TRecruit | THrRecruit, ...params: any[]): any;}// 2. 实现技术面试策略class TechnicalInterviewStrategy implements ValidateRecruitersStrategy {  validateRecruiters(recruiters: THrInterviewer[], recruit: TRecruit | THrRecruit, ..._params: any[]): any {    // 技术面试只关心前两个参数,忽略 _params    console.log("Executing TechnicalInterviewStrategy...");    return validateTechnicalInterview(recruiters, recruit);  }}// 2. 实现HR面试策略class HrInterviewStrategy implements ValidateRecruitersStrategy {  validateRecruiters(recruiters: THrInterviewer[], recruit: TRecruit | THrRecruit, ...params: any[]): any {    // HR面试需要额外的参数    console.log("Executing HrInterviewStrategy...");    const [param3, param4] = params;    return matchHrRecruiters(recruiters, recruit as THrRecruit, param3, param4);  }}// 3. 定义招聘者类别处理器,使用策略模式export const recruitersCategoryHandlers = {  TECHNICAL_INTERVIEW: {    getter: getRecruiters,    setter: new TechnicalInterviewStrategy(),  },  HR_INTERVIEW: {    getter: getHrRecruiters,    setter: new HrInterviewStrategy(),  },};// 假设的外部函数和类型function getMatchingSlots(validatedRecruiters: any, slots: any): any {  console.log("Getting matching slots with:", validatedRecruiters);  // 实际逻辑  return validatedRecruiters;}// 假设的类型 EInterviewCategorytype EInterviewCategory = 'TECHNICAL_INTERVIEW' | 'HR_INTERVIEW';// 模拟外部依赖函数function getRecruiters(): THrInterviewer[] { return [{ id: 'tech1' }]; }function getHrRecruiters(): THrInterviewer[] { return [{ id: 'hr1' }]; }function validateTechnicalInterview(recruiters: THrInterviewer[], recruit: TRecruit | THrRecruit): any {  console.log("validateTechnicalInterview called with:", recruiters, recruit);  return { validTechRecruiters: recruiters };}function matchHrRecruiters(recruiters: THrInterviewer[], recruit: THrRecruit, p3: any, p4: any): any {  console.log("matchHrRecruiters called with:", recruiters, recruit, p3, p4);  return { matchedHrRecruiters: recruiters, p3, p4 };}// --- 实际调用示例 ---// 确定面试类别和所需参数const interviewCategory: EInterviewCategory = 'HR_INTERVIEW'; // 也可以是 'TECHNICAL_INTERVIEW'const recruitData: TRecruit | THrRecruit = { id: 'someRecruitId', name: 'John Doe' }; // 模拟 recruit 数据const param3 = 'projectA';const param4 = 'teamB';const slotsWithEmail = { slot1: 'email1' }; // 模拟 slotsWithEmail// 获取初始招聘者const initialRecruiters = recruitersCategoryHandlers[interviewCategory].getter();// 调用 validateRecruitersconst validatedResult = recruitersCategoryHandlers[interviewCategory].setter.validateRecruiters(  initialRecruiters,  recruitData,  param3,  param4);// 获取匹配的槽位const matchedSlots = getMatchingSlots(validatedResult, slotsWithEmail);console.log("Final matched slots:", matchedSlots);// 另一个例子:TECHNICAL_INTERVIEWconst techInterviewCategory: EInterviewCategory = 'TECHNICAL_INTERVIEW';const techRecruitData: TRecruit = { id: 'techRecruit', name: 'Jane Smith' };const techInitialRecruiters = recruitersCategoryHandlers[techInterviewCategory].getter();const techValidatedResult = recruitersCategoryHandlers[techInterviewCategory].setter.validateRecruiters(  techInitialRecruiters,  techRecruitData,  // 对于技术面试,这里即使传入 param3, param4 也会被 TechnicalInterviewStrategy 忽略  'extraParam1', 'extraParam2');const techMatchedSlots = getMatchingSlots(techValidatedResult, slotsWithEmail);console.log("Final matched slots for tech interview:", techMatchedSlots);

注意事项与优点

灵活性和可扩展性:

新增面试类型: 如果需要添加新的面试类型(例如 PRODUCT_INTERVIEW),只需创建一个新的策略类实现 ValidateRecruitersStrategy 接口,并将其添加到 recruitersCategoryHandlers 对象中即可,无需修改现有代码。这完全符合“开闭原则”。参数变化: 即使未来某个面试类型的参数需求发生变化,也只需修改对应的策略类,而不会影响其他策略或核心调用逻辑。

代码清晰度与可维护性:

每个策略类都专注于一个特定的算法或行为,使得代码结构更加清晰,职责分离。避免了在主逻辑中出现大量的 if/else if 或 switch 语句来判断面试类型并调用不同签名的函数,降低了复杂性。

统一的调用接口:

通过 ValidateRecruitersStrategy 接口和 …params: any[],我们为所有 validateRecruiters 方法提供了一个统一的调用签名。在调用时,可以一次性传递所有可能的参数,具体的策略类会根据自身需求选择性地使用这些参数。

类型安全(TypeScript):

虽然 …params: any[] 提供了最大的灵活性,但在 TypeScript 中,可以进一步结合类型守卫或更复杂的泛型来增强类型安全性,如果不同策略的参数类型差异非常大且需要严格控制。然而,对于本例中部分参数是可选或不相关的情况,any[] 是一个实用且有效的方案。

总结

通过引入策略模式,我们成功解决了在动态选择函数时,函数参数签名不一致的挑战。这种设计模式不仅使代码更加灵活、易于扩展,而且显著提升了代码的可读性和可维护性。在面对需要根据运行时条件执行不同行为的场景时,策略模式提供了一种优雅且强大的解决方案。

以上就是动态参数下函数调用的策略模式实践的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1519107.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
利用策略模式优化不同参数函数调用的设计
上一篇 2025年12月20日 12:08:52
灵活调用不同参数签名的函数:策略模式实践指南
下一篇 2025年12月20日 12:09:04

相关推荐

  • 修复Django电商项目中AJAX过滤产品列表图片不显示问题

    在Django电商项目中,当使用AJAX动态加载过滤后的产品列表时,常遇到图片无法正常显示的问题。这通常是由于前端模板中图片加载方式(如data-setbg属性结合JavaScript库)与AJAX动态内容更新机制不兼容所致。解决方案是直接在AJAX返回的HTML中使用标准的标签来渲染图片,确保浏览…

    2026年5月10日
    700
  • Matplotlib 地图中多类型图例的创建与优化

    Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化

    本教程旨在解决matplotlib地图可视化中,如何在一个图例中同时展示颜色块(如区域分类)和自定义标记(如特定兴趣点)的问题。文章详细介绍了当传统`patch`对象无法正确显示标记时,如何利用`matplotlib.lines.line2d`创建标记图例句柄,并将其与颜色块图例句柄合并,从而生成一…

    2026年5月10日 用户投稿
    900
  • Golang JSON序列化:控制敏感字段暴露的最佳实践

    本教程探讨golang中如何高效控制结构体字段在json序列化时的可见性。当需要将包含敏感信息的结构体数组转换为json响应时,通过利用`encoding/json`包提供的结构体标签,特别是`json:”-“`,可以轻松实现对特定字段的忽略,从而避免敏感数据泄露,确保api…

    2026年5月10日
    300
  • 比特币新手教程 比特币交易平台有哪些

    比特币是一种去中心化的数字货币,基于区块链技术实现点对点交易,具有匿名性、有限发行和不可篡改等特点;新手可通过交易所购买,P2P交易获得比特币,常用平台包括Binance、OKX和Huobi;交易流程包括注册账户、实名认证、绑定支付方式、充值法币并下单购买,可选择市价单或限价单;比特币存储方式有交易…

    2026年5月10日
    000
  • c++中的SFINAE技术是什么_c++模板编程中的SFINAE原理与应用

    SFINAE 是“替换失败不是错误”的原则,指模板实例化时若参数替换导致错误,只要存在其他合法候选,编译器不报错而是继续重载决议。它用于条件启用模板、类型检测等场景,如通过 decltype 或 enable_if 控制函数重载,实现类型特征判断。尽管 C++20 引入 Concepts 简化了部分…

    2026年5月10日
    000
  • 如何让动态追加元素的类事件生效?

    如何在追加元素后使其绑定类事件生效 在页面中引入三方 JavaScript 类并通过添加相应 class 来调用事件方法是一种常见的做法。然而,如果通过 JavaScript 追加标签元素,即使添加了对应的 class,事件也可能无法生效。 为了解决这个问题,可以尝试以下步骤: 检查追加的标签是否为…

    2026年5月10日
    000
  • Golang gRPC流式请求异常处理

    在Golang的gRPC流式通信中,必须通过context.Context处理异常。应监听上下文取消或超时,及时释放资源,设置合理超时,避免连接长时间挂起,并在goroutine中通过context控制生命周期。 在使用 Golang 和 gRPC 实现流式通信时,异常处理是确保服务健壮性的关键部分…

    2026年5月10日
    000
  • Go语言mgo查询构建:深入理解bson.M与日期范围查询的正确实践

    本文旨在解决go语言mgo库中构建复杂查询时,特别是涉及嵌套`bson.m`和日期范围筛选的常见错误。我们将深入剖析`bson.m`的类型特性,解释为何直接索引`interface{}`会导致“invalid operation”错误,并提供一种推荐的、结构清晰的代码重构方案,以确保查询条件能够正确…

    2026年5月10日
    100
  • vscode上怎么运行html_vscode上运行html步骤【指南】

    首先保存文件为.html格式,再通过浏览器或Live Server插件打开预览;推荐安装Live Server实现本地服务器运行与实时刷新,提升开发体验。 在 VS Code 上运行 HTML 文件并不需要复杂的配置,只需几个简单步骤即可预览页面效果。VS Code 本身是一个代码编辑器,不直接运行…

    2026年5月10日
    100
  • RichHandler与Rich Progress集成:解决显示冲突的教程

    在使用rich库的`richhandler`进行日志输出并同时使用`progress`组件时,可能会遇到显示错乱或溢出问题。这通常是由于为`richhandler`和`progress`分别创建了独立的`console`实例导致的。解决方案是确保日志处理器和进度条组件共享同一个`console`实例…

    2026年5月10日
    300
  • 修复点击时按钮抖动:CSS垂直对齐实践

    本文探讨了在Web开发中,交互式按钮(如播放/暂停按钮)在点击时发生意外垂直位移的问题。通过分析CSS样式变化对元素布局的影响,我们发现这是由于按钮不同状态下的边框样式和内边距改变,以及默认的垂直对齐行为共同作用所致。核心解决方案是利用CSS的vertical-align属性,将其设置为middle…

    2026年5月10日
    100
  • Golang goroutine与channel调试技巧

    使用go run -race检测数据竞争,结合runtime.NumGoroutine监控协程数量,通过pprof分析阻塞调用栈,利用select超时避免永久阻塞,有效排查goroutine泄漏、死锁和数据竞争问题。 Go语言的goroutine和channel是并发编程的核心,但它们也带来了调试上…

    2026年5月10日
    000
  • 《魔兽世界》将于6月11日开启国服回归技术测试

    《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试

    《%ign%ignore_a_1%re_a_1%》官方宣布,将于6月11日开启国服回归技术测试,时间为7天,并称可以在6月内正式开服,玩家们可以访问官网下载战网客户端并预下载“巫妖王之怒”客户端,技术测试详情见下图。 WordAi WordAI是一个AI驱动的内容重写平台 53 查看详情 以上就是《…

    2026年5月10日 用户投稿
    400
  • 使用 Jupyter Notebook 进行探索性数据分析

    Jupyter Notebook通过单元格实现代码与Markdown结合,支持数据导入(pandas)、清洗(fillna)、探索(matplotlib/seaborn可视化)、统计分析(describe/corr)和特征工程,便于记录与分享分析过程。 Jupyter Notebook 是进行探索性…

    2026年5月10日
    000
  • 如何在HTML中插入表单元素_HTML表单控件与输入类型使用指南

    HTML表单通过标签构建,包含action和method属性定义数据提交目标与方式,常用input类型如text、password、email等适配不同输入需求,配合label、required、placeholder提升可用性,结合textarea、select、button等控件实现完整交互,是…

    2026年5月10日
    300
  • 前端缓存策略与JavaScript存储管理

    根据数据特性选择合适的存储方式并制定清晰的读写与清理逻辑,能显著提升前端性能;合理运用Cookie、localStorage、sessionStorage、IndexedDB及Cache API,结合缓存策略与定期清理机制,可在保证用户体验的同时避免安全与性能隐患。 前端缓存和JavaScript存…

    2026年5月10日
    200
  • HTML5网页如何实现手势操作 HTML5网页移动端交互的处理技巧

    首先利用原生touch事件实现滑动判断,再通过preventDefault解决滚动冲突,接着引入Hammer.js处理复杂手势,最后通过优化点击区域、避免事件冲突和增加视觉反馈提升体验。 在移动端浏览器中,HTML5网页可以通过触摸事件实现手势操作,提升用户体验。虽然原生JavaScript提供了基…

    2026年5月10日
    000
  • 创建指定大小并填充特定数据的Golang文件教程

    本文将介绍如何使用Golang创建一个指定大小的文件,并用特定数据填充它。我们将使用 `os` 包提供的函数来创建和截断文件,从而实现快速生成大文件的目的。示例代码展示了如何创建一个10MB的文件,并将其填充为全零数据。掌握这些方法,可以方便地在例如日志系统或磁盘队列等场景中,预先创建测试文件或初始…

    2026年5月10日
    000
  • Python命令怎样使用profile分析脚本性能 Python命令性能分析的基础教程

    使用Python的cProfile模块分析脚本性能最直接的方式是通过命令行执行python -m cProfile your_script.py,它会输出每个函数的调用次数、总耗时、累积耗时等关键指标,帮助定位性能瓶颈;为进一步分析,可将结果保存为文件python -m cProfile -o ou…

    2026年5月10日
    000
  • 如何插入查询结果数据_SQL插入Select查询结果方法

    如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法

    使用INSERT INTO…SELECT语句可高效插入数据,通过NOT EXISTS、LEFT JOIN、MERGE语句或唯一约束避免重复;表结构不一致时可通过别名、类型转换、默认值或计算字段处理;结合存储过程可提升可维护性,支持参数化与动态SQL。 将查询结果数据插入到另一个表中,可以…

    2026年5月10日 用户投稿
    400

发表回复

登录后才能评论
关注微信