[函数式编程] 为什么要柯里化?

上一篇,我们简单看了下柯里化函数是什么。

来到这里,你大概带着为什么要柯里化的问题。现在,我们一块来解答下它。

在函数式编程中,柯里化函数是为了函数组合。小函数经过柯里化后,成了可组合的"积木块",再由组合函数把它们拼装成复杂逻辑的大函数。

我们来看一个格式化用户名的例子。首先我们创建三个小函数,他们分别在用户名前添加前缀,添加后缀到用户名后面,以及把用户名转成大写。

js 复制代码
let toUpperCase = str => str.toUpperCase();
let append = (suffix, str) => str + suffix;
let prepend = (prefix, str) => prefix + str;

现在,我们组合调用他们,确实,我们格式化姓名Addy,得到结果'Hello, ADDY!'

js 复制代码
prepend('Hello,', toUpperCase(append('!', 'Addy')))

但上述代码存在2个问题

  • 如果我需要对另一个名字,比如'tom',做同样的格式化,需要再次完全地调用一遍
  • 函数嵌套式的调用方式,可读性弱

如何解决这两个问题

其实,我们可以通过柯里化函数,与函数组合工具,把这三个函数组合成一个完整的格式化函数。

下面我们写一个函数组合工具

js 复制代码
const compose = (...fns) => x => fns.reduceRight((v, fn) => fn(v), x);

组合前,柯里化这些小函数是必要的

js 复制代码
const curry = fn => (...args) =>
  args.length >= fn.length
    ? fn(...args)
    : (...next) => curry(fn)(...args, ...next);
    
toUpperCase = curry(toUpperCase);
append = curry(append);
prepend = curry(prepend);

最后,组合这三个小函数,得到一个完整的格式化函数

js 复制代码
const formatName = compose(
  append('!'),
  toUpperCase,
  prepend('Hello, ')
);

console.log(formatName('Addy')); // Hello, ADDY!
console.log(formatName('tom')); // Hello, TOM!

以上代码,由compose组合的函数们,以顺序列表的形式出现,即他们被执行的顺序,可读性强。另外,最终得到的组合函数,可以被复用。

总结

由此可见,函数式编程中,柯里化是为了组合小函数(功能)为复合功能的大函数。如果你是一个细心的读者,可能会发问,那如何让循环,条件,异步也成为可组合的函数呢?这个问题我们下回讨论。

相关推荐
世伟爱吗喽几秒前
threejs入门学习日记
前端·javascript·three.js
朝阳58114 分钟前
用 Rust + Actix-Web 打造“Hello, WebSocket!”——从握手到回声,只需 50 行代码
前端·websocket·rust
F2E_Zhangmo14 分钟前
基于cornerstone3D的dicom影像浏览器 第五章 在Displayer四个角落显示信息
开发语言·前端·javascript
slim~33 分钟前
javaweb基础第一天总结(HTML-CSS)
前端·css·html
一支鱼38 分钟前
leetcode常用解题方案总结
前端·算法·leetcode
惜.己1 小时前
针对nvm不能导致npm和node生效的解决办法
前端·npm·node.js
F2E_Zhangmo1 小时前
基于cornerstone3D的dicom影像浏览器 第二章 加载本地文件夹中的dicom文件并归档
前端·javascript·css
用户21411832636022 小时前
Nano Banana免费方案来了!Docker 一键部署 + 魔搭即开即用,小白也能玩转 AI 图像编辑
前端
Zacks_xdc2 小时前
【前端】使用Vercel部署前端项目,api转发到后端服务器
运维·服务器·前端·安全·react.js
给月亮点灯|2 小时前
Vue基础知识-脚手架开发-使用Axios发送异步请求+代理服务器解决前后端分离项目的跨域问题
前端·javascript·vue.js