mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-25 23:11:02 +08:00
PRF:20170627 An introduction to functional programming in JavaScript.md
@trnhoe 翻译的很好
This commit is contained in:
parent
202ab15df5
commit
8a7097d6b6
@ -1,77 +1,59 @@
|
||||
JavaScript 函数式编程介绍
|
||||
============================================================
|
||||
|
||||
### 探索函数式编程和通过它让你的程序更具有可读性和易于调试
|
||||
> 探索函数式编程,通过它让你的程序更具有可读性和易于调试
|
||||
|
||||
|
||||
![An introduction to functional programming in JavaScript](https://opensource.com/sites/default/files/styles/image-full-size/public/images/business/bus-rocket.png?itok=vHKRzkyW "An introduction to functional programming in JavaScript")
|
||||
![An introduction to functional programming in JavaScript](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/bus-rocket.png?itok=xQqKRMla "An introduction to functional programming in JavaScript")
|
||||
|
||||
>Image credits : Steve Jurvetson via [Flickr][80] (CC-BY-2.0)
|
||||
|
||||
当 Brendan Eich 在 1995 年创造了 JavaScript,他打算[将 Scheme 移植到浏览器里][81] 。Scheme 作为 Lisp 的方言,是一种函数式编程语言。而当 Eich 被告知新的语言应该是一种可以与 Java 相比的脚本语言后,他最终确立了一种拥有 C 风格语法的语言(也和 Java 一样),但将函数视作一等公民。Java 直到版本 8 才从技术上将函数视为一等公民,虽然你可以用匿名类来模拟它。这个特性允许 JavaScript 通过函数式范式编程。
|
||||
当 Brendan Eich 在 1995 年创造 JavaScript 时,他原本打算[将 Scheme 移植到浏览器里][81] 。Scheme 作为 Lisp 的方言,是一种函数式编程语言。而当 Eich 被告知新的语言应该是一种可以与 Java 相比的脚本语言后,他最终确立了一种拥有 C 风格语法的语言(也和 Java 一样),但将函数视作一等公民。而 Java 直到版本 8 才从技术上将函数视为一等公民,虽然你可以用匿名类来模拟它。这个特性允许 JavaScript 通过函数式范式编程。
|
||||
|
||||
JavaScript 是一个多范式语言,允许你自由地混合和使用面向对象式,过程式和函数式的范式。最近,函数式编程越来越火热。在诸如 [Angular][82] 和 [React ][83]这样的框架中,通过使用不可变数据结构实际上可以提高性能。不可变是函数式编程的核心原则,纯函数和它使得编写和调试程序变得更加容易。使用函数来代替程序的循环可以提高程序的可读性并使它更加优雅。总之,函数式编程拥有很多优点。
|
||||
JavaScript 是一个多范式语言,允许你自由地混合和使用面向对象式、过程式和函数式的编程范式。最近,函数式编程越来越火热。在诸如 [Angular][82] 和 [React][83] 这样的框架中,通过使用不可变数据结构可以切实提高性能。不可变是函数式编程的核心原则,它以及纯函数使得编写和调试程序变得更加容易。使用函数来代替程序的循环可以提高程序的可读性并使它更加优雅。总之,函数式编程拥有很多优点。
|
||||
|
||||
### 什么不是函数式编程
|
||||
|
||||
在讨论什么是函数式编程前,让我们先排除那些不属于函数式编程的东西。实际上它们是你需要丢弃的语言组件(再见,老朋友):
|
||||
|
||||
|
||||
|
||||
* 循环
|
||||
* **while**
|
||||
|
||||
* **do...while**
|
||||
|
||||
* **for**
|
||||
|
||||
* **for...of**
|
||||
|
||||
* **for...in**
|
||||
* 用 **var** 或者 **let** 来声明变量
|
||||
* 循环:
|
||||
* `while`
|
||||
* `do...while`
|
||||
* `for`
|
||||
* `for...of`
|
||||
* `for...in`
|
||||
* 用 `var` 或者 `let` 来声明变量
|
||||
* 没有返回值的函数
|
||||
* 改变对象的属性 (比如: **o.x = 5;**)
|
||||
* 改变 Array 本身的方法
|
||||
* **copyWithin**
|
||||
|
||||
* **fill**
|
||||
|
||||
* **pop**
|
||||
|
||||
* **push**
|
||||
|
||||
* **reverse**
|
||||
|
||||
* **shift**
|
||||
|
||||
* **sort**
|
||||
|
||||
* **splice**
|
||||
|
||||
* **unshift**
|
||||
* 改变 Map 本身的方法
|
||||
* **clear**
|
||||
|
||||
* **delete**
|
||||
|
||||
* **set**
|
||||
* 改变 Set 本身的方法
|
||||
* **add**
|
||||
|
||||
* **clear**
|
||||
|
||||
* **delete**
|
||||
* 改变对象的属性 (比如: `o.x = 5;`)
|
||||
* 改变数组本身的方法:
|
||||
* `copyWithin`
|
||||
* `fill`
|
||||
* `pop`
|
||||
* `push`
|
||||
* `reverse`
|
||||
* `shift`
|
||||
* `sort`
|
||||
* `splice`
|
||||
* `unshift`
|
||||
* 改变映射本身的方法:
|
||||
* `clear`
|
||||
* `delete`
|
||||
* `set`
|
||||
* 改变集合本身的方法:
|
||||
* `add`
|
||||
* `clear`
|
||||
* `delete`
|
||||
|
||||
脱离这些特性应该如何编写程序呢?这是我们将在后面探索的问题。
|
||||
|
||||
### 纯函数
|
||||
|
||||
你的程序中包含函数不一定意味着你正在进行函数式编程。函数式范式将纯函数和非纯函数区分开。鼓励你编写纯函数。纯函数必须满足下面的两个属性:
|
||||
你的程序中包含函数不一定意味着你正在进行函数式编程。函数式范式将<ruby>纯函数<rt>pure function</rt></ruby>和<ruby>非纯函数<rt>impure function</rt></ruby>区分开。鼓励你编写纯函数。纯函数必须满足下面的两个属性:
|
||||
|
||||
* **引用透明:**函数在传入相同的参数后永远返回相同的返回值。这意味着该函数不依赖于任何可变状态。
|
||||
* **Side-effect free无函数副作用:**函数不能导致任何副作用。副作用可能包括 I/O(比如 向终端或者日志文件写入),改变一个不可变的对象,对变量重新赋值等等。
|
||||
|
||||
我们来看一些例子。首先,**multiply** 就是一个纯函数的例子,它在传入相同的参数后永远返回相同的返回值,并且不会导致副作用。
|
||||
* 引用透明:函数在传入相同的参数后永远返回相同的返回值。这意味着该函数不依赖于任何可变状态。
|
||||
* 无副作用:函数不能导致任何副作用。副作用可能包括 I/O(比如向终端或者日志文件写入),改变一个不可变的对象,对变量重新赋值等等。
|
||||
|
||||
我们来看一些例子。首先,`multiply` 就是一个纯函数的例子,它在传入相同的参数后永远返回相同的返回值,并且不会导致副作用。
|
||||
|
||||
```
|
||||
function multiply(a, b) {
|
||||
@ -79,9 +61,7 @@ function multiply(a, b) {
|
||||
}
|
||||
```
|
||||
|
||||
[view raw][5][pure-function-example.js][6] hosted with ❤ by [GitHub][7]
|
||||
|
||||
下面是非纯函数的例子。**canRide** 函数依赖捕获的 **heightRequirement** 变量。被捕获的变量不一定导致一个函数是非纯函数,除非它是一个可变的变量(或者可以被重新赋值)。这种情况下使用 **let** 来声明这个变量,意味着可以对它重新赋值。**multiply** 函数是非纯函数,因为它会导致在 console 上输出。
|
||||
下面是非纯函数的例子。`canRide` 函数依赖捕获的 `heightRequirement` 变量。被捕获的变量不一定导致一个函数是非纯函数,除非它是一个可变的变量(或者可以被重新赋值)。这种情况下使用 `let` 来声明这个变量,意味着可以对它重新赋值。`multiply` 函数是非纯函数,因为它会导致在 console 上输出。
|
||||
|
||||
```
|
||||
let heightRequirement = 46;
|
||||
@ -98,36 +78,27 @@ function multiply(a, b) {
|
||||
}
|
||||
```
|
||||
|
||||
|
||||
[view raw][8][impure-functions.js][9] hosted with ❤ by [GitHub][10]
|
||||
|
||||
下面的列表包含着 JavaScript 内置的非纯函数。你可以指出它们不满足两个属性中的哪个吗?
|
||||
|
||||
* **console.log**
|
||||
* `console.log`
|
||||
* `element.addEventListener`
|
||||
* `Math.random`
|
||||
* `Date.now`
|
||||
* `$.ajax` (这里 `$` 代表你使用的 Ajax 库)
|
||||
|
||||
* **element.addEventListener**
|
||||
|
||||
* **Math.random**
|
||||
|
||||
* **Date.now**
|
||||
|
||||
* **$.ajax** (**$** 代表你使用的 Ajax 库)
|
||||
|
||||
理想的程序中所有的函数都是纯函数,但是从上面的函数列表可以看出,任何有意义的程序都将包含非纯函数。大多时候我们需要进行 AJAX 调用,检查当前日期或者获取一个随机数。一个好的经验法则是遵循80/20规则:函数中有80%应该是纯函数,剩下的20%的必要性将不可避免地是非纯函数。
|
||||
理想的程序中所有的函数都是纯函数,但是从上面的函数列表可以看出,任何有意义的程序都将包含非纯函数。大多时候我们需要进行 AJAX 调用,检查当前日期或者获取一个随机数。一个好的经验法则是遵循 80/20 规则:函数中有 80% 应该是纯函数,剩下的 20% 的必要性将不可避免地是非纯函数。
|
||||
|
||||
使用纯函数有几个优点:
|
||||
|
||||
* 它们很容易导出和调试,因为它们不依赖于可变的状态。
|
||||
|
||||
* 返回值可以被缓存或者“记忆”来避免以后重复计算。
|
||||
|
||||
* 它们很容易测试,因为没有需要模拟(mock)的依赖(比如日志,AJAX,数据库等等)。
|
||||
|
||||
你编写或者使用的函数返回空(换句话说它没有返回值),那代表它是非纯函数。
|
||||
|
||||
### 不变性
|
||||
|
||||
让我们回到捕获变量的概念上。来看看 **canRide** 函数。我们认为它是一个非纯函数,因为 **heightRequirement** 变量可以被重新赋值。下面是一个构造出来的例子来说明如何用不可预测的值来对它重新赋值。
|
||||
让我们回到捕获变量的概念上。来看看 `canRide` 函数。我们认为它是一个非纯函数,因为 `heightRequirement` 变量可以被重新赋值。下面是一个构造出来的例子来说明如何用不可预测的值来对它重新赋值。
|
||||
|
||||
```
|
||||
let heightRequirement = 46;
|
||||
@ -146,10 +117,7 @@ const mySonsHeight = 47;
|
||||
setInterval(() => console.log(canRide(mySonsHeight)), 500);
|
||||
```
|
||||
|
||||
|
||||
[view raw][11][mutable-state.js][12] hosted with ❤ by [GitHub][13]
|
||||
|
||||
我要再次强调被捕获的变量不一定会使函数成为非纯函数。我们可以通过只是简单地改变 **heightRequirement** 的声明方式来使 **canRide** 函数成为纯函数。
|
||||
我要再次强调被捕获的变量不一定会使函数成为非纯函数。我们可以通过只是简单地改变 `heightRequirement` 的声明方式来使 `canRide` 函数成为纯函数。
|
||||
|
||||
```
|
||||
const heightRequirement = 46;
|
||||
@ -159,10 +127,7 @@ function canRide(height) {
|
||||
}
|
||||
```
|
||||
|
||||
|
||||
[view raw][14][immutable-state.js][15] hosted with ❤ by [GitHub][16]
|
||||
|
||||
通过用 **const** 来声明变量意味着它不能被再次赋值。如果尝试对它重新赋值,运行时引擎将抛出错误;那么,如果用对象来代替数字来存储所有的“常量”怎么样?
|
||||
通过用 `const` 来声明变量意味着它不能被再次赋值。如果尝试对它重新赋值,运行时引擎将抛出错误;那么,如果用对象来代替数字来存储所有的“常量”怎么样?
|
||||
|
||||
```
|
||||
const constants = {
|
||||
@ -175,58 +140,53 @@ function canRide(height) {
|
||||
}
|
||||
```
|
||||
|
||||
|
||||
[view raw][17][captured-mutable-object.js][18] hosted with ❤ by [GitHub][19]
|
||||
|
||||
我们用了 **const** ,所以这个变量不能被重新赋值,但是还有一个问题:这个对象可以被改变。下面的代码展示了,为了真正使其不可变,你不仅需要防止它被重新赋值,你也需要不可变的数据结构。JavaScript 语言提供了 **Object.freeze** 方法来阻止对象被改变。
|
||||
我们用了 `const` ,所以这个变量不能被重新赋值,但是还有一个问题:这个对象可以被改变。下面的代码展示了,为了真正使其不可变,你不仅需要防止它被重新赋值,你也需要不可变的数据结构。JavaScript 语言提供了 `Object.freeze` 方法来阻止对象被改变。
|
||||
|
||||
```
|
||||
'use strict';
|
||||
|
||||
// CASE 1: The object is mutable and the variable can be reassigned.对象的属性是可变的,并且变量可以被再次赋值。
|
||||
// CASE 1: 对象的属性是可变的,并且变量可以被再次赋值。
|
||||
let o1 = { foo: 'bar' };
|
||||
|
||||
// Mutate the object改变对象的属性
|
||||
// 改变对象的属性
|
||||
o1.foo = 'something different';
|
||||
|
||||
// Reassign the variable对变量再次赋值
|
||||
// 对变量再次赋值
|
||||
o1 = { message: "I'm a completely new object" };
|
||||
|
||||
|
||||
// CASE 2: The object is still mutable but the variable cannot be reassigned.对象的属性还是可变的,但是变量不能被再次赋值。
|
||||
// CASE 2: 对象的属性还是可变的,但是变量不能被再次赋值。
|
||||
const o2 = { foo: 'baz' };
|
||||
|
||||
// Can still mutate the object
|
||||
// 仍然能改变对象
|
||||
o2.foo = 'Something different, yet again';
|
||||
|
||||
// Cannot reassign the variable不能对变量再次赋值
|
||||
// 不能对变量再次赋值
|
||||
// o2 = { message: 'I will cause an error if you uncomment me' }; // Error!
|
||||
|
||||
|
||||
// CASE 3: The object is immutable but the variable can be reassigned.对象的属性是不可变的,但是变量可以被再次赋值。
|
||||
// CASE 3: 对象的属性是不可变的,但是变量可以被再次赋值。
|
||||
let o3 = Object.freeze({ foo: "Can't mutate me" });
|
||||
|
||||
// Cannot mutate the object不能改变对象的属性
|
||||
// 不能改变对象的属性
|
||||
// o3.foo = 'Come on, uncomment me. I dare ya!'; // Error!
|
||||
|
||||
// Can still reassign the variable还是可以对变量再次赋值
|
||||
// 还是可以对变量再次赋值
|
||||
o3 = { message: "I'm some other object, and I'm even mutable -- so take that!" };
|
||||
|
||||
|
||||
// CASE 4: The object is immutable and the variable cannot be reassigned. This is what we want!!!!!!!!对象的属性是不可变的,并且变量不能被再次赋值。这是我们想要的!!!!!!!!
|
||||
// CASE 4: 对象的属性是不可变的,并且变量不能被再次赋值。这是我们想要的!!!!!!!!
|
||||
const o4 = Object.freeze({ foo: 'never going to change me' });
|
||||
|
||||
// Cannot mutate the object不能改变对象的属性
|
||||
// 不能改变对象的属性
|
||||
// o4.foo = 'talk to the hand' // Error!
|
||||
|
||||
// Cannot reassign the variable不能对变量再次赋值
|
||||
// 不能对变量再次赋值
|
||||
// o4 = { message: "ain't gonna happen, sorry" }; // Error
|
||||
```
|
||||
|
||||
|
||||
[view raw][20][immutability-vs-reassignment.js][21] hosted with ❤ by [GitHub][22]
|
||||
|
||||
不变性适用于所有的数据结构,包括数组,映射和集合。它意味着不能调用例如 **Array.prototype.push** 等会导致本身改变的方法,因为它会改变已经存在的数组。可以通过创建一个含有原来元素和新加元素的新数组,代替将新元素加入一个已经存在的数组。其实所有会导致数组本身被修改的方法都可以通过一个返回修改好的新数组的函数代替。**这里一段mutator的翻译大概有些啰嗦。**
|
||||
不变性适用于所有的数据结构,包括数组、映射和集合。它意味着不能调用例如 `Array.prototype.push` 等会导致本身改变的方法,因为它会改变已经存在的数组。可以通过创建一个含有原来元素和新加元素的新数组,而不是将新元素加入一个已经存在的数组。其实所有会导致数组本身被修改的方法都可以通过一个返回修改好的新数组的函数代替。
|
||||
|
||||
```
|
||||
'use strict';
|
||||
@ -251,16 +211,13 @@ const f = R.sort(myCompareFunction, a); // R = Ramda
|
||||
// Instead of: a.reverse();
|
||||
const g = R.reverse(a); // R = Ramda
|
||||
|
||||
// Exercise for the reader留给读者的练习:
|
||||
// 留给读者的练习:
|
||||
// copyWithin
|
||||
// fill
|
||||
// splice
|
||||
```
|
||||
|
||||
|
||||
[view raw][23][array-mutator-method-replacement.js][24] hosted with ❤ by [GitHub][25]
|
||||
|
||||
[**Map**][84] 和 [**Set**][85] 也很相似。可以通过返回一个新的修改好的 Map 或者 Set 来代替使用会修改其本身的函数。
|
||||
[映射][84] 和 [集合][85] 也很相似。可以通过返回一个新的修改好的映射或者集合来代替使用会修改其本身的函数。
|
||||
|
||||
```
|
||||
const map = new Map([
|
||||
@ -279,9 +236,6 @@ const map3 = new Map([...map].filter(([key]) => key !== 1));
|
||||
const map4 = new Map();
|
||||
```
|
||||
|
||||
|
||||
[view raw][26][map-mutator-method-replacement.js][27] hosted with ❤ by [GitHub][28]
|
||||
|
||||
```
|
||||
const set = new Set(['A', 'B', 'C']);
|
||||
|
||||
@ -295,19 +249,15 @@ const set3 = new Set([...set].filter(key => key !== 'B'));
|
||||
const set4 = new Set();
|
||||
```
|
||||
|
||||
我想提一句如果你在使用 TypeScript(我非常喜欢 TypeScript),你可以用 `Readonly<T>`、`ReadonlyArray<T>`、`ReadonlyMap<K, V>` 和 `ReadonlySet<T>` 接口来在编译期检查你是否尝试更改这些对象,有则抛出编译错误。如果在对一个对象字面量或者数组调用 `Object.freeze`,编译器会自动推断它是只读的。由于映射和集合在其内部表达,所以在这些数据结构上调用 `Object.freeze` 不起作用。但是你可以轻松地告诉编译器它们是只读的变量。
|
||||
|
||||
[view raw][29][set-mutator-method-replacement.js][30] hosted with ❤ by [GitHub][31]
|
||||
|
||||
我想提一句如果你在使用 TypeScript(我非常喜欢TypeScript),你可以用 **Readonly<T>**, **ReadonlyArray<T>**, **ReadonlyMap<K, V>**, 和 **ReadonlySet<T>** 接口来在编译期检查你是否尝试更改这些对象,有则抛出编译错误。如果在对一个对象字面量或者数组调用 **Object.freeze**,编译器会自动推断它是只读的。由于映射和集合在其内部表达,所以在这些数据结构上调用 **Object.freeze** 不起作用。但是你可以轻松地告诉编译器它们是只读的变量。
|
||||
|
||||
### [typescript-readonly.png][32]
|
||||
|
||||
![TypeScript Readonly Interfaces](https://opensource.com/sites/default/files/u128651/typescript-readonly.png "TypeScript Readonly Interfaces")
|
||||
|
||||
|
||||
TypeScript read-only interfaces
|
||||
*TypeScript 只读接口*
|
||||
|
||||
好,所以我们可以通过创建新的对象来代替修改原来的对象,但是这样不会导致性能损失吗?当然会。确保在你自己的应用中做了性能测试。如果你需要提高性能,可以考虑使用 [Immutable.js][86]。Immutable.js 用[持久的数据结构][91] 实现了[链表][87],,[堆栈][88], [映射][89],[集合][90]和其他数据结构。使用了和比如 Clojure 和 Scala 的函数式语言中相同的技术。
|
||||
好,所以我们可以通过创建新的对象来代替修改原来的对象,但是这样不会导致性能损失吗?当然会。确保在你自己的应用中做了性能测试。如果你需要提高性能,可以考虑使用 [Immutable.js][86]。Immutable.js 用[持久的数据结构][91] 实现了[链表][87]、[堆栈][88]、[映射][89]、[集合][90]和其他数据结构。使用了如同 Clojure 和 Scala 这样的函数式语言中相同的技术。
|
||||
|
||||
```
|
||||
// Use in place of `[]`.
|
||||
@ -338,17 +288,14 @@ console.log([...set1]); // [1, 2, 3, 4]
|
||||
console.log([...set2]); // [1, 2, 3, 4, 5]
|
||||
```
|
||||
|
||||
[view raw][33][immutable-js-demo.js][34] hosted with ❤ by [GitHub][35]
|
||||
### 函数组合
|
||||
|
||||
### 函数组合(不确定是这个叫法
|
||||
|
||||
好,你准备好听解答了吗?**f ∘ g **读作 **“函数 f 和函数 g 组合”(不确定 没搜到中文资料**。它的定义是 **(f ∘ g)(x) = f(g(x))**,你也可以认为 **f ∘ g** 是一个单独的函数,它将调用函数 **g** 的结果作为参数传给函数 **f**。注意这些函数是从右向左以此调用的,先执行 **g**,接下来执行 **f**。
|
||||
记不记得在中学时我们学过一些像 `(f ∘ g)(x)` 的东西?你那时可能想,“我什么时候会用到这些?”,好了,现在就用到了。你准备好了吗?`f ∘ g`读作 “函数 f 和函数 g 组合”。对它的理解有两种等价的方式,如等式所示: `(f ∘ g)(x) = f(g(x))`。你可以认为 `f ∘ g` 是一个单独的函数,或者视作将调用函数 `g` 的结果作为参数传给函数 `f`。注意这些函数是从右向左依次调用的,先执行 `g`,接下来执行 `f`。
|
||||
|
||||
关于函数组合的几个要点:
|
||||
|
||||
1. 我们可以组合任意数量的函数(不仅限于 2 个)。
|
||||
|
||||
2. 组合函数的一个方式是简单地把一个函数的输出作为下一个函数的输入。(比如 **f(g(x))**))
|
||||
2. 组合函数的一个方式是简单地把一个函数的输出作为下一个函数的输入(比如 `f(g(x))`)。
|
||||
|
||||
```
|
||||
// h(x) = x + 1
|
||||
@ -374,9 +321,7 @@ const y = f(g(h(1)));
|
||||
console.log(y); // '4'
|
||||
```
|
||||
|
||||
[view raw][36][function-composition-basic.js][37] hosted with ❤ by [GitHub][38]
|
||||
|
||||
[Ramda][92] 和 [lodash][93] 之类的库提供了更优雅的方式来组合函数。我们可以在更多的在数学意义上处理函数组合,而不是简单地将一个函数的返回值传递给下一个函数。我们可以创建一个由这些函数组成的单一复合函数(就是,**(f ∘ g)(x)**)。
|
||||
[Ramda][92] 和 [lodash][93] 之类的库提供了更优雅的方式来组合函数。我们可以在更多的在数学意义上处理函数组合,而不是简单地将一个函数的返回值传递给下一个函数。我们可以创建一个由这些函数组成的单一复合函数(就是 `(f ∘ g)(x)`)。
|
||||
|
||||
```
|
||||
// h(x) = x + 1
|
||||
@ -406,17 +351,15 @@ const y = composite(1);
|
||||
console.log(y); // '4'
|
||||
```
|
||||
|
||||
[view raw][39][function-composition-elegant.js][40] hosted with ❤ by [GitHub][41]
|
||||
|
||||
我们可以在 JavaScript 中组合函数。接下来呢?好,如果你已经入门了函数式编程,理想中你的程序将只有函数的组合。代码里没有循环(**for**, **for...of**, **for...in**, **while**, **do**),基本没有。你说但是那不可能。并不是这样。我们下面的两个话题是:递归和高阶函数。
|
||||
好了,我们可以在 JavaScript 中组合函数了。接下来呢?好,如果你已经入门了函数式编程,理想中你的程序将只有函数的组合。代码里没有循环(`for`, `for...of`, `for...in`, `while`, `do`),基本没有。你可能觉得那是不可能的。并不是这样。我们下面的两个话题是:递归和高阶函数。
|
||||
|
||||
### 递归
|
||||
|
||||
假设你想实现一个计算数字的阶乘的函数。 让我们回顾一下数学中阶乘的定义:
|
||||
|
||||
**n! = n * (n-1) * (n-2) * ... * 1**.
|
||||
`n! = n * (n-1) * (n-2) * ... * 1`.
|
||||
|
||||
**n!** 是从 **n** 到 **1** 的所有整数的乘积。我们可以编写一个循环轻松地计算出结果。
|
||||
`n!` 是从 `n` 到 `1` 的所有整数的乘积。我们可以编写一个循环轻松地计算出结果。
|
||||
|
||||
```
|
||||
function iterativeFactorial(n) {
|
||||
@ -428,16 +371,13 @@ function iterativeFactorial(n) {
|
||||
}
|
||||
```
|
||||
|
||||
注意 `product` 和 `i` 都在循环中被反复重新赋值。这是解决这个问题的标准过程式方法。如何用函数式的方法解决这个问题呢?我们需要消除循环,确保没有变量被重新赋值。递归是函数式程序员的最有力的工具之一。递归需要我们将整体问题分解为类似整体问题的子问题。
|
||||
|
||||
[view raw][42][iterative-factorial.js][43] hosted with ❤ by [GitHub][44]
|
||||
计算阶乘是一个很好的例子,为了计算 `n!` 我们需要将 n 乘以所有比它小的正整数。它的意思就相当于:
|
||||
|
||||
注意 **product** 和 **i** 都在循环中被反复重新赋值。这是解决这个问题的标准过程式方法。如何用函数式的方法解决这个问题呢?我们需要消除循环,确保没有变量被重新赋值。递归是函数式程序员的最有力的工具之一。递归需要我们将整体问题分解为类似整体问题的子问题。
|
||||
`n! = n * (n-1)!`
|
||||
|
||||
计算阶乘是一个很好的例子,为了计算 **n!** 我们需要将 n 乘以所有比它小的正整数。它的意思就相当于:
|
||||
|
||||
**n! = n * (n-1)!**
|
||||
|
||||
A-ha!我们发现了一个类似于整个问题 **n!** 的子问题来解决 **(n-1)!**。还有一个需要注意的地方就是基础条件。基础条件告诉我们何时停止递归。 如果我们没有基础条件,那么递归将永远持续。 实际上,如果有太多的递归调用,程序会抛出一个堆栈溢出错误。A-HA!
|
||||
啊哈!我们发现了一个解决 `(n-1)!` 的子问题,它类似于整个问题 `n!`。还有一个需要注意的地方就是基础条件。基础条件告诉我们何时停止递归。 如果我们没有基础条件,那么递归将永远持续。 实际上,如果有太多的递归调用,程序会抛出一个堆栈溢出错误。啊哈!
|
||||
|
||||
```
|
||||
function recursiveFactorial(n) {
|
||||
@ -449,28 +389,21 @@ function recursiveFactorial(n) {
|
||||
}
|
||||
```
|
||||
|
||||
|
||||
[view raw][45][recursive-factorial.js][46] hosted with ❤ by [GitHub][47]
|
||||
|
||||
然后我们来计算 **recursiveFactorial(20000)** 因为...,为什么不呢?当我们这样做的时候,我们得到了这个结果:
|
||||
|
||||
### [stack-overflow.png][48]
|
||||
然后我们来计算 `recursiveFactorial(20000)` 因为……,为什么不呢?当我们这样做的时候,我们得到了这个结果:
|
||||
|
||||
![Stack overflow error](https://opensource.com/sites/default/files/u128651/stack-overflow.png "Stack overflow error")
|
||||
|
||||
堆栈溢出错误
|
||||
*堆栈溢出错误*
|
||||
|
||||
这里发生了什么?我们得到一个堆栈溢出错误!这不是无穷的递归导致的。我们已经处理了基础条件(**n === 0** 的情况)。那是因为浏览器的堆栈大小是有限的,而我们的代码使用了越过了这个大小的堆栈。每次对 **recursiveFactorial** 的调用导致了新的帧被压入堆栈中,就像一个盒子压在另一个盒子上。每当 **recursiveFactorial** 被调用,一个新的盒子被放在最上面。下图展示了在计算 **recursiveFactorial(3)** 时堆栈的样子。注意在真实的堆栈中,堆栈顶部的帧将存储在执行完成后应该返回的内存地址,但是我选择用变量 **r** 来表示返回值,因为 JavaScript 开发者一般不需要考虑内存地址。
|
||||
|
||||
### [stack-frames.png][49]
|
||||
这里发生了什么?我们得到一个堆栈溢出错误!这不是无穷的递归导致的。我们已经处理了基础条件(`n === 0` 的情况)。那是因为浏览器的堆栈大小是有限的,而我们的代码使用了越过了这个大小的堆栈。每次对 `recursiveFactorial` 的调用导致了新的帧被压入堆栈中,就像一个盒子压在另一个盒子上。每当 `recursiveFactorial` 被调用,一个新的盒子被放在最上面。下图展示了在计算 `recursiveFactorial(3)` 时堆栈的样子。注意在真实的堆栈中,堆栈顶部的帧将存储在执行完成后应该返回的内存地址,但是我选择用变量 `r` 来表示返回值,因为 JavaScript 开发者一般不需要考虑内存地址。
|
||||
|
||||
![The stack for recursively calculating 3! (three factorial)](https://opensource.com/sites/default/files/u128651/stack-frames.png "The stack for recursively calculating 3! (three factorial)")
|
||||
|
||||
递归计算 3! 的堆栈(三次乘法)
|
||||
*递归计算 3! 的堆栈(三次乘法)*
|
||||
|
||||
你可能会想象当计算 **n = 20000** 时堆栈会更高。我们可以做些什么优化它吗?当然可以。作为 **ES2015** (又名 **ES6**) 标准的一部分,有一个优化用来解决这个问题。它被称作_尾调用优化_。它使得浏览器删除或者忽略堆栈帧,当递归函数做的最后一件事是调用自己并返回结果的时候。实际上,这个优化对于相互递归函数也是有效的,但是为了简单起见,我们还是来看单递归函数。
|
||||
你可能会想象当计算 `n = 20000` 时堆栈会更高。我们可以做些什么优化它吗?当然可以。作为 ES2015 (又名 ES6) 标准的一部分,有一个优化用来解决这个问题。它被称作<ruby>尾调用优化<rt>proper tail calls optimization</rt></ruby>(PTC)。当递归函数做的最后一件事是调用自己并返回结果的时候,它使得浏览器删除或者忽略堆栈帧。实际上,这个优化对于相互递归函数也是有效的,但是为了简单起见,我们还是来看单一递归函数。
|
||||
|
||||
你可能会注意到,在递归函数调用之后,还要进行一次额外的计算(**n * r**)。那意味着浏览器不能通过 PTC 来优化递归;然而,我们可以通过重写函数使最后一步变成递归调用以便优化。一个窍门是将中间结果(在这里是**乘积**)作为参数传递给函数。
|
||||
你可能会注意到,在递归函数调用之后,还要进行一次额外的计算(`n * r`)。那意味着浏览器不能通过 PTC 来优化递归;然而,我们可以通过重写函数使最后一步变成递归调用以便优化。一个窍门是将中间结果(在这里是 `product`)作为参数传递给函数。
|
||||
|
||||
```
|
||||
'use strict';
|
||||
@ -484,30 +417,23 @@ function factorial(n, product = 1) {
|
||||
}
|
||||
```
|
||||
|
||||
|
||||
[view raw][50][factorial-tail-recursion.js][51] hosted with ❤ by [GitHub][52]
|
||||
|
||||
让我们来看看优化后的计算 **factorial(3)** 时的堆栈。如下图所示,堆栈不会增长到超过两层。原因是我们把必要的信息都传到了递归函数中(比如**乘积**)。所以,在**乘积**被更新后,浏览器可以丢弃掉堆栈中原先的帧。你可以在图中看到每次最上面的帧下沉变成了底部的帧,原先底部的帧被丢弃,因为不再需要它了。
|
||||
|
||||
### [optimized-stack-frames.png][53]
|
||||
让我们来看看优化后的计算 `factorial(3)` 时的堆栈。如下图所示,堆栈不会增长到超过两层。原因是我们把必要的信息都传到了递归函数中(比如 `product`)。所以,在 `product` 被更新后,浏览器可以丢弃掉堆栈中原先的帧。你可以在图中看到每次最上面的帧下沉变成了底部的帧,原先底部的帧被丢弃,因为不再需要它了。
|
||||
|
||||
![The optimized stack for recursively calculating 3! (three factorial) using PTC](https://opensource.com/sites/default/files/u128651/optimized-stack-frames.png "The optimized stack for recursively calculating 3! (three factorial) using PTC")
|
||||
|
||||
递归计算 3! 的堆栈(三次乘法)使用 PTC
|
||||
*递归计算 3! 的堆栈(三次乘法)使用 PTC*
|
||||
|
||||
现在选一个浏览器运行吧,假设你在使用 Safari,你会得到**无穷大**(它是比在 JavaScript 中能表达的最大值更大的数)。但是我们没有得到堆栈溢出错误,那很不错!现在在其他的浏览器中呢怎么样呢?Safari 可能现在乃至将来是实现 PTC 的唯一一个浏览器。看看下面的兼容性表格:
|
||||
|
||||
### [ptc-compatibility.png][54]
|
||||
现在选一个浏览器运行吧,假设你在使用 Safari,你会得到 `Infinity`(它是比在 JavaScript 中能表达的最大值更大的数)。但是我们没有得到堆栈溢出错误,那很不错!现在在其他的浏览器中呢怎么样呢?Safari 可能现在乃至将来是实现 PTC 的唯一一个浏览器。看看下面的兼容性表格:
|
||||
|
||||
![PTC compatibility](https://opensource.com/sites/default/files/u128651/ptc-compatibility.png "PTC compatibility")
|
||||
|
||||
PTC 兼容性
|
||||
*PTC 兼容性*
|
||||
|
||||
其他浏览器提出了一种被称作[语法级尾调用][95](STC)的竞争标准。"Syntactic"意味着你需要用新的语法来标识你想要执行尾递归优化的函数。,即使浏览器还没有广泛支持,但是把你的递归函数写成支持尾递归优化的样子还是一个好主意。
|
||||
其他浏览器提出了一种被称作<ruby>[语法级尾调用][95]<rt>syntactic tail calls</rt></ruby>(STC)的竞争标准。“语法级”意味着你需要用新的语法来标识你想要执行尾递归优化的函数。即使浏览器还没有广泛支持,但是把你的递归函数写成支持尾递归优化的样子还是一个好主意。
|
||||
|
||||
### 高阶函数
|
||||
|
||||
我们已经知道 JavaScript 将函数视作一等公民,可以把函数像其他值一样传递。所以,把一个函数传给另一个函数也很常见。我们也可以让函数返回一个函数。就是它!我们有高阶函数。你可能已经很熟悉几个在 **Array.prototype** 中的高阶函数。比如 **filter**,**map**,和 **reduce** 就在其中。对高阶函数的一种理解是:它是接受(一般会调用)一个回调函数参数的函数。让我们来看看一些内置的高阶函数的例子:
|
||||
我们已经知道 JavaScript 将函数视作一等公民,可以把函数像其他值一样传递。所以,把一个函数传给另一个函数也很常见。我们也可以让函数返回一个函数。就是它!我们有高阶函数。你可能已经很熟悉几个在 `Array.prototype` 中的高阶函数。比如 `filter`、`map` 和 `reduce` 就在其中。对高阶函数的一种理解是:它是接受(一般会调用)一个回调函数参数的函数。让我们来看看一些内置的高阶函数的例子:
|
||||
|
||||
```
|
||||
const vehicles = [
|
||||
@ -531,10 +457,7 @@ const averageSUVPrice = vehicles
|
||||
console.log(averageSUVPrice); // 33399
|
||||
```
|
||||
|
||||
|
||||
[view raw][55][built-in-higher-order-functions.js][56] hosted with ❤ by [GitHub][57]
|
||||
|
||||
注意我们在一个数组对象上调用其方法,这是面向对象编程的特性。如果我们想要更函数式一些,我们可以用 Rmmda 或者 lodash/fp 提供的函数。注意如果我们使用 **R.compose** 的话,需要倒转函数的顺序,因为它从右向左依次调用函数(从底向上);然而,如果我们想从左向右调用函数就像上面的例子,我们可以用 **R.pipe**。下面两个例子用了 Rmmda。注意 Rmmda 有一个 **mean** 函数用来代替 **reduce** 。
|
||||
注意我们在一个数组对象上调用其方法,这是面向对象编程的特性。如果我们想要更函数式一些,我们可以用 Rmmda 或者 lodash/fp 提供的函数。注意如果我们使用 `R.compose` 的话,需要倒转函数的顺序,因为它从右向左依次调用函数(从底向上);然而,如果我们想从左向右调用函数就像上面的例子,我们可以用 `R.pipe`。下面两个例子用了 Rmmda。注意 Rmmda 有一个 `mean` 函数用来代替 `reduce` 。
|
||||
|
||||
```
|
||||
const vehicles = [
|
||||
@ -569,14 +492,11 @@ const averageSUVPrice2 = R.compose(
|
||||
console.log(averageSUVPrice2); // 33399
|
||||
```
|
||||
|
||||
|
||||
[view raw][58][composing-higher-order-functions.js][59] hosted with ❤ by [GitHub][60]
|
||||
|
||||
使用函数式方法的优点是清楚地分开了数据(**车辆**)和逻辑(函数 **filter**,**map** 和 **reduce**)面向对象的代码相比之下把数据和函数用以方法的对象的形式混合在了一起。
|
||||
使用函数式方法的优点是清楚地分开了数据(`vehicles`)和逻辑(函数 `filter`,`map` 和 `reduce`)。面向对象的代码相比之下把数据和函数用以方法的对象的形式混合在了一起。
|
||||
|
||||
### 柯里化
|
||||
|
||||
不规范地说,柯里化是把一个接受 **n** 个参数的函数变成 **n** 个每个接受单个参数的函数的过程。函数的 **arity** 是它接受参数的个数。接受一个参数的函数是 **unary**,两个的是 **binary**,三个的是 **ternary**,**n**个的是 **n-ary**。那么,我们可以把柯里化定义成将一个 **n-ary** 函数转换成 **n** 个 unary 函数的过程。让我们通过简单的例子开始,一个计算两个向量点积的函数。回忆一下线性代数,两个向量 **[a, b, c]** 和 **[x, y, z]** 的点积是 **ax + by + cz**。
|
||||
不规范地说,<ruby>柯里化<rt>currying</rt></ruby>是把一个接受 `n` 个参数的函数变成 `n` 个每个接受单个参数的函数的过程。函数的 `arity` 是它接受参数的个数。接受一个参数的函数是 `unary`,两个的是 `binary`,三个的是 `ternary`,`n` 个的是 `n-ary`。那么,我们可以把柯里化定义成将一个 `n-ary` 函数转换成 `n` 个 `unary` 函数的过程。让我们通过简单的例子开始,一个计算两个向量点积的函数。回忆一下线性代数,两个向量 `[a, b, c]` 和 `[x, y, z]` 的点积是 `ax + by + cz`。
|
||||
|
||||
```
|
||||
function dot(vector1, vector2) {
|
||||
@ -589,10 +509,7 @@ const v2 = [4, -2, -1];
|
||||
console.log(dot(v1, v2)); // 1(4) + 3(-2) + (-5)(-1) = 4 - 6 + 5 = 3
|
||||
```
|
||||
|
||||
|
||||
[view raw][61][dot-product.js][62] hosted with ❤ by [GitHub][63]
|
||||
|
||||
**dot** 函数是 binary 因为它接受两个参数;然而我们可以将它手动转换成两个 unary 函数,就像下面的例子。注意 **curriedDot** 是一个 unary 函数,它接受一个向量并返回另一个接受第二个向量的 unary 函数。
|
||||
`dot` 函数是 binary,因为它接受两个参数;然而我们可以将它手动转换成两个 unary 函数,就像下面的例子。注意 `curriedDot` 是一个 unary 函数,它接受一个向量并返回另一个接受第二个向量的 unary 函数。
|
||||
|
||||
```
|
||||
function curriedDot(vector1) {
|
||||
@ -609,9 +526,6 @@ console.log(sumElements([1, 3, -5])); // -1
|
||||
console.log(sumElements([4, -2, -1])); // 1
|
||||
```
|
||||
|
||||
|
||||
[view raw][64][manual-currying.js][65] hosted with ❤ by [GitHub][66]
|
||||
|
||||
很幸运,我们不需要把每一个函数都手动转换成柯里化以后的形式。[Ramda][96] 和 [lodash][97] 等库可以为我们做这些工作。实际上,它们是柯里化的混合形式。你既可以每次传递一个参数,也可以像原来一样一次传递所有参数。
|
||||
|
||||
```
|
||||
@ -634,8 +548,6 @@ console.log(sumElements(v2)); // 1
|
||||
console.log(curriedDot(v1, v2)); // 3
|
||||
```
|
||||
|
||||
[view raw][67][fancy-currying.js][68] hosted with ❤ by [GitHub][69]
|
||||
|
||||
Ramda 和 lodash 都允许你“跳过”一些变量之后再指定它们。它们使用置位符来做这些工作。因为点积的计算可以交换两项。传入向量的顺序不影响结果。让我们换一个例子来阐述如何使用一个置位符。Ramda 使用双下划线作为其置位符。
|
||||
|
||||
```
|
||||
@ -657,11 +569,9 @@ console.log(result);
|
||||
// 3: French Hens
|
||||
```
|
||||
|
||||
[view raw][70][currying-placeholder.js][71] hosted with ❤ by [GitHub][72]
|
||||
在我们结束探讨柯里化之前最后的议题是<ruby>偏函数应用<rt>partial application</rt></ruby>。偏函数应用和柯里化经常同时出场,尽管它们实际上是不同的概念。一个柯里化的函数还是柯里化的函数,即使没有给它任何参数。偏函数应用,另一方面是仅仅给一个函数传递部分参数而不是所有参数。柯里化是偏函数应用常用的方法之一,但是不是唯一的。
|
||||
|
||||
在我们结束探讨柯里化之前最后的议题是偏函数应用。偏函数应用和柯里化经常同时出场,尽管它们实际上是不同的概念。一个柯里化的函数还是柯里化的函数,即使没有给它任何参数。偏函数应用,另一方面是仅仅给一个函数传递部分而不是所有参数。柯里化是偏函数应用常用的方法之一,但是不是唯一的。
|
||||
|
||||
JavaScript 拥有一个内置机制可以不依靠柯里化来偏函数应用。那就是 [**function.prototype.bind**][98] 方法。这个方法的一个特殊之处在于,它要求你将 **this** 作为第一个参数传入。 如果你不进行面向对象编程,那么你可以通过传入 **null** 来忽略 **this**。
|
||||
JavaScript 拥有一个内置机制可以不依靠柯里化来做偏函数应用。那就是 [function.prototype.bind][98] 方法。这个方法的一个特殊之处在于,它要求你将 `this` 作为第一个参数传入。 如果你不进行面向对象编程,那么你可以通过传入 `null` 来忽略 `this`。
|
||||
|
||||
```
|
||||
1function giveMe3(item1, item2, item3) {
|
||||
@ -682,30 +592,25 @@ console.log(result);
|
||||
// 3: scissors
|
||||
```
|
||||
|
||||
|
||||
[view raw][73][partial-application-using-bind.js][74] hosted with ❤ by [GitHub][75]
|
||||
|
||||
### 总结
|
||||
|
||||
我希望你享受探索 JavaScript 中函数式编程的过程。对一些人来说,它可能是一个全新的编程范式,但我希望你能尝试它。你会发现你的程序更易于阅读和调试。不变性还将允许你优化 Angular 和 React 的性能。
|
||||
|
||||
_这篇文章基于 Matt 在 OpenWest 的演讲 [JavaScript the Good-er Parts][77]. [OpenWest][78] ~~将~~在 6/12-15 ,2017 在 Salt Lake City, Utah 举行。_
|
||||
_这篇文章基于 Matt 在 OpenWest 的演讲 [JavaScript the Good-er Parts][77]. [OpenWest][78] ~~将~~在 6/12-15 ,2017 在 Salt Lake City, Utah 举行。_
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
作者简介:
|
||||
|
||||
Matt Banz - Matt graduated from the University of Utah with a degree in mathematics in May of 2008. One month later he landed a job as a web developer, and he's been loving it ever since! In 2013, he earned a Master of Computer Science degree from North Carolina State University. He has taught courses in web development for LDS Business College and the Davis School District Community Education program. He is currently a Senior Front-End Developer for Motorola Solutions.
|
||||
|
||||
Matt Banz - Matt 于 2008 年五月在犹他大学获得了数学学位毕业。一个月后他得到了一份 web 开发者的工作,他从那时起就爱上了它!在 2013 年,他在北卡罗莱纳州立大学获得了计算机科学硕士学位。他在 LDS 商学院和戴维斯学区社区教育计划教授 Web 课程。他现在是就职于 Motorola Solutions 公司的高级前端开发者。
|
||||
|
||||
--------------
|
||||
|
||||
via: https://opensource.com/article/17/6/functional-javascript
|
||||
|
||||
作者:[Matt Banz ][a]
|
||||
作者:[Matt Banz][a]
|
||||
译者:[trnhoe](https://github.com/trnhoe)
|
||||
校对:[校对者ID](https://github.com/校对者ID)
|
||||
校对:[wxy](https://github.com/wxy)
|
||||
|
||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user