【第2845期】JS 函数式概念: 管道 和 组合

前言

函数式编程的概念。今日前端早读课文章由 @前端小智翻译授权分享。

正文从这开始~~

函数管道和组合是函数式编程中的概念,当然也可以在 JavaScript 中实现 -- 因为它是一种多范式的编程语言,让我们快速深入了解这个概念。

这个概念就是按照一定的顺序执行多个函数,并将一个函数的结果传递给下一个函数。

你可以像这样做得很难看:

 function1(function2(function3(initialArg)))

或者使用函数组合:

 compose(function3, function2, function1)(initialArg);

或功能管道:

 pipe(function1, function2, function3)(initialArg);

简而言之,组合和管道几乎是一样的,唯一的区别是执行顺序;如果函数从左到右执行,就是管道,另一方面,如果函数从右到左执行,就叫组合。

一个更准确的定义是。"在函数式编程中,compose 是将较小的单元(我们的函数)组合成更复杂的东西(你猜对了,是另一个函数)的机制"。

下面是一个管道函数的例子。

 const pipe = (...functions) => (value) => {
return functions.reduce((currentValue, currentFunction) => {
return currentFunction(currentValue);
}, value);
};

让我们来补充一些这方面的见解。

基础知识

  • 我们需要收集 N 多的函数

  • 同时选择一个参数

  • 以链式方式执行它们,将收到的参数传递给将被执行的第一个函数

  • 调用下一个函数,加入第一个函数的结果作为参数。

  • 继续对数组中的每个函数做同样的操作。

 /* destructuring to unpack our array of functions into functions */
const pipe = (...functions) =>
/* value is the received argument */
(value) => {
/* reduce helps by doing the iteration over all functions stacking the result */
return functions.reduce((currentValue, currentFunction) => {
/* we return the current function, sending the current value to it */
return currentFunction(currentValue);
}, value);
};

我们已经知道,箭头函数如果只返回一条语句,就不需要括号,也不需要返回标签,所以我们可以通过这样写来减少键盘的点击次数。

 const pipe = (...functions) => (input) => functions.reduce((chain, func) => func(chain), input);

如何使用

 const pipe = (...fns) => (input) => fns.reduce((chain, func) => func(chain), input);

const sum = (...args) => args.flat(1).reduce((x, y) => x + y);

const square = (val) => val*val;

pipe(sum, square)([3, 5]); // 64

记住,第一个函数是左边的那个(Pipe),所以 3+5=8,8 的平方是 64。我们的测试很顺利,一切似乎都很正常,但如果要用链式 async 函数呢?

异步函数上的管道

我在这方面的一个用例是有一个中间件来处理客户端和网关之间的请求,过程总是相同的(做请求,错误处理,挑选响应中的数据,处理响应以烹制一些数据,等等等等),所以让它看起来像一个魅力。

 export default async function handler(req, res) {
switch (req.method) {
case 'GET':
return pipeAsync(provide, parseData, answer)(req.headers);
/*
...
*/

让我们看看如何在 Javascript 和 Typescript 中处理异步函数管道。

JS 版
 export const pipeAsync =
(...fns) =>
(input) =>
fns.reduce((chain, func) => chain.then(func), Promise.resolve(input));

添加了 JSDoc 类型,使其更容易理解(我猜)。

 /**
* Applies Function piping to an array of async Functions.
* @param {Promise<Function>[]} fns
* @returns {Function}
*/

export const pipeAsync =
(...fns) =>
(/** @type {any} */ input) =>
fns.reduce((/** @type {Promise<Function>} */ chain, /** @type {Function | Promise<Function> | any} */ func) => chain.then(func), Promise.resolve(input));
TS 版
 export const pipeAsync: any =
(...fns: Promise<Function>[]) =>
(input: any) =>
fns.reduce((chain: Promise<Function>, func: Function | Promise<Function> | any) => chain.then(func), Promise.resolve(input));

这样一来,它对异步和非异步函数都有效,所以它比上面的例子更胜一筹。

你可能想知道函数的组成是什么,所以让我们来看看。

函数组合

如果你喜欢从右到左调用这些函数,你只需要将 reduce 改为 redureRight,就可以了。让我们看看用函数组成的异步方式。

 export const composeAsync =
(...fns) =>
(input) =>
fns.reduceRight((chain, func) => chain.then(func), Promise.resolve(input));

回到上面的例子,让我们复制同样的内容,但要有构图。

如何使用
 const compose = (...fns) => (input) => fns.reduceRight((chain, func) => func(chain), input);

const sum = (...args) => args.flat(1).reduce((x, y) => x + y);

const square = (val) => val*val;

compose(square, sum)([3, 5]); // 64

请注意,我们颠倒了函数的顺序,以保持与帖子顶部的例子一致。

现在,sum(位于最右边的位置)将被首先调用,因此 3+5=8,然后 8 的平方是 64。

关于本文
译者:@前端小智
译文:https://segmentfault.com/a/1190000043278104
作者:@JoelBonetR
原文:https://dev.to/joelbonetr/js-functional-concepts-pipe-and-compose-1mho


关于函数组合】相关推荐,欢迎读者自荐投稿,前端早读课等你来。+v:zhgb_f2er

【第2674期】Javascript函数组合:有什么大不了的?

这期前端早读课

对你有帮助,帮”  “一下,

期待下一期,帮” 在看”一下 。

相关推荐

  • 哔哩哔哩⼤数据建设之路—实时DQC篇
  • 为内存塞不下Transformer犯愁?OpenAI应用AI研究负责人写了份指南
  • AI正在让很多行业的红利消失
  • Hadoop、Spark等5种大数据框架对比,你的项目该用哪种?
  • 谈谈ChatGPT 背后的核心技术论文
  • 美国击落中国无人飞艇;上海平均月薪10605元;湖南高速多车相撞致16死66伤;黄光裕已不是国美零售控股股东.....|酷玩日爆
  • 同学的爸妈是丁克?| 每日一冷
  • 坑了乐视两千万的熊猫,曾经却是中国工业摇篮
  • 全球生育率下降的原因,在于社保福利对生孩子的惩罚性
  • 外交部就美方宣称击落中国无人飞艇发表声明;西方对俄罗斯石油产品限价;韩国一艘渔船倾覆致9人失踪 | 每日大新闻
  • 韩国禁毒失败启示录:扫毒不彻底,等于彻底不扫毒
  • 历经风波,elementary OS 7正式发布
  • 解读ChatGPT背后的技术重点:RLHF、IFT、CoT、红蓝对抗
  • 面试必问的异步顺序问题,用 Performance 轻松理清
  • “我靠淘宝无货源副业,月入过万”:聪明的人,从不挣辛苦钱!!
  • 为什么特斯拉打响新能源汽车降价第一枪?
  • 全网汇总Java笔试面试题
  • 从零开发一个Python项目,手势识别!
  • Caffeine Cache-高性能Java本地缓存组件
  • 世界最“反内卷”的小镇,4点下班,全镇只有一栋楼,200多人住进去,半年出一次屋…