├── .gitignore
├── README.md
├── advance.md
├── beginer.md
├── examples
├── news
│ ├── db
│ │ └── news.json
│ ├── fp-example-news.xml
│ ├── index.html
│ ├── package.json
│ ├── src
│ │ ├── index.js
│ │ └── utils.js
│ └── webpack.config.js
└── todoList
│ ├── index.html
│ ├── package.json
│ ├── src
│ ├── index.css
│ └── index.jsx
│ └── webpack.config.js
└── fp-composition-law.xml
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules/
2 | .DS_Store
3 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # 函数式编程系列教程
2 |
3 | > javascript函数式编程指南。
4 |
5 | 写这个文章的原因在于函数式编程的思想非常先进,其天生的可预测性(也可以说是可测试), 更细粒度的代码(逻辑)重用,以及天生支持并行等特点, 已经被也业内越来越多的人认可。由于其很高的学习门槛导致大多人并不了解它,或者只是知道概念,并不能够理解并运用函数式编程,因此写下了这篇文章
6 | 。当然本人也不是函数式编程大师,如有错误,请斧正。
7 |
8 | ## 目录
9 |
10 | [函数式编程-入门](https://github.com/azl397985856/functional-programming/blob/master/beginer.md)
11 |
12 |
13 | [函数式编程-进阶](https://github.com/azl397985856/functional-programming/blob/master/advance.md)
14 |
15 |
16 | [函数式编程-精通]()
17 |
--------------------------------------------------------------------------------
/advance.md:
--------------------------------------------------------------------------------
1 | # 函数式编程-进阶
2 | 本文是函数式编程系列教程的第二篇。如果你对函数式编程不了解,可以看下之前的[函数式编程-基础篇](https://github.com/azl397985856/functional-programming/blob/master/beginer.md)。
3 | ## Hindley-Milner
4 | 开始讲接下面的内容之前,我们引入一个类型签名系统HM。它的功能非常强大,我们可以通过它知道函数做了什么,而不需要将代码读一遍。
5 | Hindley-Milner是一个经典的类型系统,他在函数式编程中的功能非常强大。
6 | 尽管你也可以在非函数式编程中使用,但是总感觉少了点什么。
7 | 类型签名不但可以用于编译时检测,而且还可以当作文档。
8 | HM是一个非常复杂的系统,如果大家想要学习的话,可以自己去查阅相关资料。
9 | 这里只是简单介绍下。如下是一个简单的例子:
10 |
11 | ```js
12 | // head :: [a] -> (a Null)
13 | function head(arr) {
14 | return arr && arr.length ? arr[0] : null;
15 | }
16 |
17 | head(["gone"]); // "gone"
18 |
19 | ```
20 |
21 | 可以看出HM由两部分组成, part one :: part two ,前面是函数的名字,后面是函数的签名。
22 | 那么head的签名意思就是接受一个任意类型的Array,返回一个该类型的元素或者Null.
23 |
24 |
25 | 我们来看一个更加复杂的例子:
26 | ```js
27 | // map :: (a -> b) -> [a] -> [b]
28 | var map = curry(function(f, xs){
29 | return xs.map(f);
30 | });
31 |
32 | ```
33 |
34 | 它表达的是map这个函数接受一个a到b的一个函数, 然后接受一个任意类型数组,最后返回另外一个任意类型的数组。
35 |
36 | 我们再来看下reduce的:
37 | ```js
38 | // reduce :: (b -> a -> b) -> b -> [a] -> b
39 | var reduce = curry(function(f, x, xs){
40 | return xs.reduce(f, x);
41 | });
42 |
43 | ```
44 |
45 | 上面表达的是reduce,接受一个b -> a -> b 这样的函数作为参数,然后成为了另外一个函数,
46 | 这个函数接受类型b为参数,然后返回一个函数,这个函数又接受[a]作为参数,最终返回b.
47 |
48 | > a b 虽然任意的类型,但是同一个类型签名中a类型和a类型一定是相同的,这点需要注意。
49 |
50 | 更多关于类型签名记号的信息,可以查看[类型签名记号](https://github.com/fantasyland/fantasy-land#type-signature-notation)
51 |
52 | 大家好好消化一下,再往后看哦~
53 |
54 | ## 再谈组合
55 |
56 | 前面我们介绍了函数组合。那么这部分我们进一步来讨论一下函数组合。
57 |
58 | 下面举一个原生JS的例子:
59 | ```js
60 | // 假设addOne和mutiFive已经在别的地方定义
61 | // 事实上,我们已经在上一节中写过这两个函数了。
62 | [1,2,3,4]
63 | .map(addOne) // [2,3,4,5]
64 | .map(multiFive)// [10, 15, 20, 25]
65 | ```
66 | 很好,我们似乎也实现了函数组合的效果。只不过我们的风格变成了链式调用。
67 |
68 | 上面的代码使用我们上一节介绍的知识,写出来应该是`compose(multiFive, addOne)`(正如我们上一节介绍的那样)。
69 |
70 | 但如果是这样的呢?
71 |
72 | ```js
73 | [1,2,3,4]
74 | .reduce(add) // 15
75 | .map(multiFive) // TypeError: xxx map is not a function
76 | ```
77 | 实际项目中,我们不只会用到数组,我们会用到很多其他类型。这个时候我们是否可以以这种map链式调用的方式实现函数组合呢?
78 |
79 | 我们是否可以使用一种通用的方式操作所有的数据类型呢?如果有了这种通用的操作方式,是否就可以实现上面的效果了呢?
80 |
81 | ### map
82 | 我们先来看下数组是如何实现链式调用的组合效果的。
83 |
84 | 数组能够通过map实现链式调用,本质上是因为其作为一个包装对象,可以通过map这样的一个高阶函数操作其内部的数据。
85 |
86 | 那么我们是否可以封装一个包装对象,使得所有数据类型都可以通过某种方式操作呢?
87 | 而这种操作,我们习惯称之为map。
88 |
89 |
90 | ### 容器
91 | 我们来实现一个Box
92 |
93 | ```js
94 | function Box(value) {
95 | this.__value = value;
96 | }
97 |
98 | Box.prototype.of = (value) => new Box(value);
99 |
100 | Box.prototype.map = function(fn) { return Box.of(fn(this.__value)) } // 箭头函数会丢失this到Box的指向
101 | ```
102 |
103 | JS中的Array也是一种BOX。 它同样提供了一种操作对象的方式map。
104 | 在函数式编程中,我们称这样的BOX为functor(函子)。
105 |
106 |
107 | 函数式编程中有很多functor, 比如option,future等,
108 | 这样我们操作的所有的数据就可以链式调用了(因为它们实现了相同的契约-map)。
109 | 实际上functor并不仅仅是盒子这么简单,它提供了一致的操作数据的方式,对外提供了简洁的api,对调用者隐藏了具体细节。
110 |
111 | 如果我们需要处理异步呢?如果我们需要处理错误(其中又会涉及到分支处理)呢?
112 |
113 | 答案就是借助刚才讲到的容器。
114 |
115 | 可以说函数式编程对异步操作是很简单的。通过前面讲到的compose,我们可以将多个函数像多米诺骨牌一样组合起来,然后按照一定的顺序执行,即使他们是异步的。
116 |
117 | > 我们需要将之前的compose做一些小改变以适应异步操作
118 |
119 | ### compose Promsie
120 | 回忆一下compose的写法:
121 |
122 | ```js
123 |
124 | function compose(...funcs) {
125 | if (funcs.length === 0) {
126 | return arg => arg
127 | }
128 |
129 | if (funcs.length === 1) {
130 | return funcs[0]
131 | }
132 |
133 | return funcs.reduce((a, b) => (...args) => a(b(...args)))
134 | }
135 | ```
136 |
137 | 以compose promise为例:
138 | ```js
139 | function composePromise(...funcs) {
140 | if (funcs.length === 0) {
141 | return arg => arg
142 | }
143 |
144 | if (funcs.length === 1) {
145 | return funcs[0]
146 | }
147 |
148 | return funcs.reduce((a, b) => (...args) => Promise.resolve(...args).then(b).then(a))
149 | }
150 |
151 | function setTimeoutPromise(number, delay) {
152 | return new Promise((resolve, reject) => {
153 | setTimeout(() => {
154 | console.log(number);
155 | resolve()
156 | }, delay)
157 | })
158 | }
159 |
160 | const setTimeoutPromiseWithOne = curry(setTimeoutPromise)(1)
161 | const setTimeoutPromiseWithTwo = curry(setTimeoutPromise)(2)
162 | composePromise(setTimeoutPromiseWithTwo, setTimeoutPromiseWithOne)(2000) // 2s先输出1,4s之后输出2
163 |
164 | ```
165 | 可以看到我们可以将异步操作给组合起来。
166 | 我们可以将函数柯里化,实现Promise.all的效果。
167 |
168 | 可以说函数式编程对异步操作是很友好的,只是需要做一些小小的”把戏“。
169 |
170 | 另外我们常常碰到的问题就是异常处理,异常处理怎么看都很难是纯函数,那么如何
171 | 以纯的方式处理异常呢? 异常处理必然要伴随着分支处理,函数式编程又是如何处理分支处理(像if/else)呢?毕竟这些在我们的日常开发中非常常见,我们接下来解释。
172 | 我们如何处理呢?
173 | ## 分支处理-Either容器
174 | 答案还是借助容器,先来看下函数式 如何优雅实现if/else,我们需要学习一个新的容器,
175 | 我们称之为Either.
176 |
177 |
178 | ```js
179 |
180 | // either做的事情就是在有value的时候将构造一个Right容器,
181 | // 否则构造一个Left容器,Left调用map操作数据的时候,不会发生任何事情
182 | // 我们可以通过定制化msg,定制一些诸如错误信息之类的
183 | var Either = function(msg, value) {
184 | if (value) return Right.of(value);
185 | return Left.of(msg)
186 | }
187 | var Left = function(x) {
188 | this.__value = x;
189 | }
190 |
191 | Left.of = function(x) {
192 | return new Left(x);
193 | }
194 |
195 | Left.prototype.map = function(f) {
196 | return this; // 什么都不做
197 | }
198 |
199 | var Right = function(x) {
200 | this.__value = x;
201 | }
202 |
203 | Right.of = function(x) {
204 | return new Right(x);
205 | }
206 |
207 | Right.prototype.map = function(f) {
208 | return Right.of(f(this.__value));
209 | }
210 |
211 | ```
212 |
213 | 我们可以用Either来实现一下上面的异常处理。
214 |
215 | ```js
216 | function readFile(filename) {
217 | return Either('filename shouldn't be empty!', filename)
218 | }
219 |
220 | readFile() // Left {__value: "filename shouldn't be empty!"}
221 | readFile('./test.txt') // Right {__value: "./test.txt"}
222 | ```
223 | 我们没有使用throw error,而是以一种更加温和更加纯粹的方式处理。
224 |
225 |
226 | ## 异步处理-Task容器
227 | 那么如何处理异步呢? 没错,我们再来引入一个容器,叫Task。
228 | Task的实现稍微复杂,在这里不再实现,大家可以自行查阅详细信息,如果你不想查,
229 | 那么你可以暂时把它看成Promise(两者相似,但也有不一样的地方)。
230 |
231 | 异步处理是不纯的,是有副作用的,我们可以直接通过HM看到哪些函数是不纯粹的。
232 |
233 | 正如Martin Odersky 在 [Google Groups Scala debate](https://groups.google.com/forum/#!topic/scala-debate/xYlUlQAnkmE%5B251-275%5D)
234 | 阐述的一样。
235 |
236 | > The IO monad does not make a function pure. It just makes it obvious that it’s impure
237 |
238 | 拿http获取用户信息为例:
239 |
240 | ```js
241 | // fetchUserInfo :: Number -> Task(Error, JSON)
242 | function fetchUserInfo(id) {
243 | return new Task(function(reject, result) {
244 | httpGet(`/user/${id}`, function(err, data) {
245 | err ? reject(err) : result(data);
246 | });
247 | });
248 | };
249 | // 假设data是{ name: 'lucifer', id: 1001, sex: 'male'}
250 | fetchUserInfo(1001).map(getName).map(toUpperCase); // LUCIFER
251 | ```
252 | 是不是觉得task和Promise很像,map就像then。
253 |
254 |
255 | 我们再来一个更加复杂的例子,
256 | 我们需要读取本地文件,然后根据文件读取的内容去请求另外一个服务的数据。以下例子摘自[mostly-adequate-guide](https://github.com/MostlyAdequate/mostly-adequate-guide)
257 |
258 | 我稍加了修改。
259 |
260 | ```js
261 | // upload :: String -> (String -> a) -> Void
262 | var upload = function(filename, callback) {
263 | if(!filename) {
264 | throw new Error("filename should't be empty!");
265 | } else {
266 | readFile(filename, function(err, data) {
267 | if(err) throw err;
268 | httpPost(data, function(err, result) {
269 | if(err) throw err;
270 | callback(result);
271 | });
272 | });
273 | }
274 | }
275 |
276 | ```
277 |
278 | 上面的代码如何用函数式编程重构一下,前提当然是readFile, httpPost也是函数式的写法喽。
279 |
280 |
281 | ```js
282 | // readFile :: Filename -> Either String (Future Error String)
283 | // httpPost :: String -> Future Error JSON
284 |
285 | // upload :: String -> Either String (Future Error JSON)
286 | var upload = compose(map(map(httpPost('/uploads'))), readFile);
287 |
288 | ```
289 |
290 | 从类型签名可以看出,我们预防了三个错误——readFile 使用 Either 来验证输入(或许还有确保文件名存在);readFile 在读取文件的时候可能会出错,错误通过 readFile 的 Future 表示;文件上传可能会因为各种各样的原因出错,错误通过 httpPost 的 Future 表示。我们就这么随意地使用 chain 实现了两个嵌套的、有序的异步执行动作。
291 | 所有这些操作都是在一个线性流中完成的,是完完全全纯的、声明式的代码,是可以等式推导(equational reasoning)并拥有可靠特性(reliable properties)的代码。我们没有被迫使用不必要甚至令人困惑的变量名,我们的 upload 函数符合通用接口而不是特定的一次性接口。
292 |
293 | 为了帮助大家理解,这里附一下函数式的readFile和httpPost的实现。
294 |
295 | 函数式实现:
296 | ```js
297 | // readFile :: Filename -> Either String (Future Error String)
298 | function readFile(filename) {
299 | return Either('filename shouldn\'t be empty!', new Task((resolve, reject) => {
300 | fs.readFile(filename, (err, data) => {
301 | if (err) return reject(err);
302 | resolve(data);
303 | }
304 | )}
305 | ))
306 | }
307 |
308 | function http(method, url, data) {
309 | return new Task((resolve, reject) => {
310 | http[method](url, {data}, (err, data) => {
311 | if (err) return reject(err);
312 | resolve(data);
313 | })
314 | })
315 | }
316 | // httpPost :: String -> Future Error JSON
317 | var httpPost = curry(http('post', '/path');
318 |
319 | ```
320 |
321 | 可以看出我们的代码重用性非常强。
322 | 类似的函数式编程中还有很多容器,
323 | 但是本质都是一样的,思想也是相通的,
324 | 大家可以自行查阅相关资料。
325 |
326 | ## 总结
327 | 这一节我们介绍了如何在函数式中使用容器处理分支逻辑,如何处理异步情况,以及如何加二者的嵌套结合起来。
328 | 对于不同的地方我们用到了不同的容器,函数式世界中有很多很多容器。
329 |
330 | 最后通过一个例子,让大家更直观地感受到函数式编程对逻辑的抽象,使得代码天生复用性,可读性都很强。
331 | 不过理解和熟练使用会比较困难, 还是那句话,重要的是改变思路,用函数式思维去编程。
332 |
333 | 下一节我们来介绍monad,applicative,以及一些类型推导,集合方面的知识。
334 | ## 参考
335 | [from-callback-to-future-functor-monad](https://hackernoon.com/from-callback-to-future-functor-monad-6c86d9c16cb5)
336 |
--------------------------------------------------------------------------------
/beginer.md:
--------------------------------------------------------------------------------
1 | # 函数式编程-入门
2 | 本篇文章是我的函数式编程教程中的第一篇文章,目的是让
3 | 大家对函数式编程有一个概念,为后面学习函数式编程提供基础。
4 |
5 | ## 介绍
6 | 函数式编程不过是相比于传统的面向过程和面向对象的一种新的编程方法而已。
7 |
8 | 他的出现并不是为了取代谁,也不会取代。他的出现只是为了改善传统的
9 | 编程方法中存在的问题。典型的一个作用就是就是限制系统的副作用。然而函数式编程
10 | 并不是用来消灭副作用,这是不切实际的。他的存在只是将副作用限制到某一个很小的范围。
11 | 使我们的程序的大部分逻辑都是纯粹的。
12 |
13 | 然而函数式编程离不开数学,你可以从函数式编程中看到数学之美。
14 | 你在初中高中大学用到的很多公式理论在这里都适用,很神奇,不是嘛?
15 | 让我们开始学习吧~
16 | ## 函数
17 | 一个常见的误解就是函数式编程中的函数指的就是用函数写代码。
18 | 比如JavaScript就是function。 这种理解是完全错误的。
19 |
20 | 然而函数式编程中的函数其实指的是数学中的函数。
21 | 让我们来回忆一下初中数学。
22 | ### 数学中的函数
23 | 我们来看下数学中的函数的定义:
24 |
25 | > 给定一个数集A,假设其中的元素为x。现对A中的元素x施加对应法则f,记作f(x),得到另一数集B。
26 | 假设B中的元素为y。则y与x之间的等量关系可以用y=f(x)表示。我们把这个关系式就叫函数关系式,简称函数。
27 |
28 | 函数概念含有三个要素:定义域A、值域C和对应法则f。其中核心是对应法则f,它是函数关系的本质特征。
29 |
30 |
31 | 函数式编程中的函数正是数学中的函数,而JavaScript中的函数是其超集。
32 | 这也就是为什么很多数学公式理论在函数式编程中都适用的原因。
33 | ### 纯函数
34 | 上面介绍了数学中的函数,我们已经知道了函数式编程中的函数指的正是数据中的函数。
35 | 那么数学中的函数在JavaScript中又是什么呢?
36 |
37 | 数学中的函数指的Javascript中的纯函数。
38 |
39 | > 这种说法不准确,因此纯函数并不是Javascript中的术语,这里只是方便解释。
40 |
41 | 那么什么是纯函数?
42 |
43 | > 纯函数就是给定输入,输出总是相同的函数。
44 |
45 | 我们在学习初中数学中的函数的时候,学过`定义域中的一个元素在值域有且仅有一个对应的值`。
46 | 这个其实和纯函数的定义是一致的。
47 |
48 | 纯函数的好处就是无副作用,不管我是执行一次,还是一百次,结果总是一样的。
49 | 那么这有什么用呢?
50 |
51 | - 可预测(也可以叫可测试)
52 |
53 | 一个很明显的好处就是可预测,即根据输入就可以知道输出。
54 | 利用这个特性,我们就可以很容易的去断言输出,也就更容易测试。
55 |
56 | redux声称是可预测的状态管理容器, 其可预测正是归功于纯函数的特性。
57 | redux中的reducer被要求是一个纯函数,所有的状态变化都经过reducer这个
58 | 纯函数去完成。 这样应用的状态(准确地讲是redux的store)就变得可预测,
59 | 也就方便测试。
60 |
61 | - 可缓存
62 |
63 | 由于给定输入,输出总是一定的。那么我们就可以将函数结果缓存起来,当
64 | `后续`调用的时候就可以直接从缓存中拿,避免了重新执行的开销。
65 | 这在大运算中是非常重要的。
66 |
67 | 很多函数式编程库都实现了memorize方法,我们拿ramda为例,
68 | 如下是ramda的官方文档对memorize的代码演示:
69 |
70 | ```js
71 | let count = 0;
72 | const factorial = R.memoize(n => {
73 | count += 1;
74 | return R.product(R.range(1, n + 1));
75 | });
76 | factorial(5); //=> 120
77 | factorial(5); //=> 120
78 | factorial(5); //=> 120
79 | count; //=> 1
80 | ```
81 |
82 | - 可并行
83 |
84 | 函数式编程由于其纯函数的特性,是天然支持并行的。
85 | 因为其不会因为时间的改变而导致函数的执行出现改变。
86 | 换句话说,在函数式编程中,时间并不是自变量。
87 |
88 | ## 一等公民
89 | 函数式编程的本质是将函数当作一等公民。
90 |
91 | 在理解这个概念以前,我们先来看下高阶函数。
92 | ### 高阶函数
93 | 高阶函数要么是以函数作为参数,要么以函数作为返回值,要么兼而有之。
94 |
95 | 一个简单的例子:
96 |
97 | ```js
98 | // 常见写法
99 | function add(x) {
100 | return y => x + y;
101 | }
102 |
103 | // 箭头函数写法
104 | const add = x => y => x + y;
105 |
106 | const addOne = add(1);
107 | addOne(2); // 3
108 | ```
109 | > 为了保持代码的简洁性,后面都采用箭头函数的形式书写。
110 |
111 | 然后js里面的map,sort,reduce等都是地地道道的高阶函数。
112 | 可以说我们一直在使用高阶函数,只是我们并不知情。就好像我们一直在使用闭包,
113 | 却可能并不知道闭包的概念一样。
114 |
115 | 事实上,js中的事件也是高阶函数。
116 |
117 | ```js
118 | // 将函数作为参数传入
119 | document.addEventListener("click", e => e);
120 | ```
121 | 高阶函数大大提高了代码的抽象能力,进而提高了代码的复用率。
122 | 通过传入不同的函数进而实现不同的效果,毫不夸张地讲,这种
123 | 抽象非常强大。理解高阶函数对于理解函数式编程至关重要,他是函数式编程中的基石。
124 | ## 柯里化
125 | 柯里化(Currying)是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,
126 | 并且返回接受余下的参数且返回结果的新函数的技术。
127 |
128 | 听起来比较拗口,让我们通过一个例子来看下。
129 | ```js
130 | // 取出对象里面的属性, 简单起见,省略了校验逻辑
131 | const get = key => obj => obj[key]
132 | const getId = get("id")
133 | const getName = get("name")
134 |
135 | const data = {
136 | id: 1,
137 | name: 'lucifer'
138 | }
139 | getId(data); // 1
140 | getName(data); // lucifer
141 | ```
142 |
143 | 上面的例子非常简洁,我们称之为pointfree风格。
144 | 让我们再看下非柯里化版本
145 |
146 | ```js
147 | const getByKey = (key, obj) => obj[key]
148 |
149 | // 繁琐
150 | const getId = data => getByKey("id", data);
151 | const getName = data => getByKey("name", data);
152 |
153 | const data = {
154 | id: 1,
155 | name: 'lucifer'
156 | }
157 |
158 | getId(data); // 1
159 | getName(data); // lucifer
160 | ```
161 |
162 | 由于非柯里化版本必须提供所有参数才能执行。
163 | 因此构造出的getId和getName必须提供data。
164 | 可能这个例子还不太明显,我们会在后面将pointfree部分举一个更复杂的例子。
165 |
166 | 其实我们前面将高阶函数部分举的add的例子已经是柯里化了。
167 | add本身接受两个参数,然后通过柯里化的方式将其编程接受一个参数的函数,执行它,
168 | 你得到了一个接受一个参数的值,再次执行,才会返回相加的结果。
169 |
170 | 纯正的函数式编程中,所有的函数都只有一个参数。
171 | 这样再js中就会写成这样`getId('id')(data)`。我们必须写下两个括号,
172 | 这是由于js本身的语法决定的,对于scala,haskell等函数式语言,不存在这样的事情。
173 | 因此我们通常使用一些函数式编程库帮我们简化成`getId('id', data)`的写法。
174 | 当然这并不以为着getId就不柯里化了。这只是一个语法糖而已。
175 |
176 | 一个简单的柯里化的js实现如下:
177 |
178 | ```js
179 | function curry(fn) {
180 | return function inner(...args) {
181 | if(args.length < fn.length) {
182 | return function(...innerArgs) {
183 | return inner.apply(this, args.concat(innerArgs))
184 | }
185 | }
186 | return fn.apply(this, args)
187 | }
188 | }
189 |
190 | ```
191 |
192 | ## 组合
193 | 通过柯里化我们将多个参数的函数变成只接受一个参数的高阶函数。
194 | 本节我们通过函数组合,将不同的函数组合形成各种不同的新的函数。
195 |
196 | 函数组合就是将函数串联起来执行,将多个函数组合起来,一个函数的输出结果是另一个函数的输入参数,一旦第一个函数开始执行,就会像多米诺骨牌一样推导执行了。
197 |
198 | 举个例子:
199 | ```js
200 | const add = x => y => x + y;
201 | const addOne = add(1);
202 | const plus = x => y => x * y;
203 | const plusFive = plus(5);
204 |
205 | const addOneAndPlusFive = compose(plusFive, addOne)
206 |
207 | addOneAndPlusFive(3); // 20
208 | ```
209 |
210 | 上面的例子很简单,就是将addOne 和 plusFive 这两个函数给组合起来,形成了一个新的函数而已。
211 | 类似的我们可以组合创造出无数的方法来。
212 |
213 | 如下是redux的compose的实现,可以帮助大家理解它是怎么运作的。
214 |
215 | ```js
216 | function compose(...funcs) {
217 | if (funcs.length === 0) {
218 | return arg => arg
219 | }
220 |
221 | if (funcs.length === 1) {
222 | return funcs[0]
223 | }
224 |
225 | return funcs.reduce((a, b) => (...args) => a(b(...args)))
226 | }
227 | ```
228 | 如上使用reduce方法将函数累计到一起,形成一个新的函数。
229 | ## Pointfree
230 | 终于到了文章的重点了。
231 |
232 | 其实前面已经提到了pointfree了。我们通过curry已经实现了pointfree风格。
233 | 但是这并不是全部。事实上要实现pointfree除了借助curry,还需要将可变化的数据放在最后(data goes last)。
234 |
235 | 前面的例子不足以解释这一点,我们举一个稍微复杂且贴近实际的例子:
236 |
237 | 如下是后端返回的数据格式
238 | ```json
239 | [{
240 | "user": "lucifer",
241 | "posts": [
242 | { "title": "fun fun function", "contents": "..." },
243 | { "title": "time slice", "contents": "..." }
244 | ]
245 | }, {
246 | "user": "lucifer",
247 | "posts": [
248 | { "title": "babel", "contents": "..." },
249 | { "title": "webpack", "contents": "..." }
250 | ]
251 | }, {
252 | "user": "karl",
253 | "posts": [
254 | { "title": "ramda", "contents": "..." },
255 | { "title": "lodash", "contents": "..." }
256 | ]
257 | }]
258 |
259 | ```
260 | 我们需要做的就是找到所有lucifer的文章,并将其title打印出来。
261 |
262 | 非pointfree的写法:
263 |
264 | ```js
265 | fetch(url)
266 | .then(JSON.parse)
267 | .then(datas => datas.filter(data => data.user === 'lucifer'))
268 | .then(datas => datas.map(data => data.posts).reduce((pre, next) => pre.concat(next)))
269 | .then(posts => posts.map(post => post.title))
270 |
271 | ```
272 | pointfree的写法:
273 |
274 | ```js
275 | // 这里为了演示,并没有将fetch这样的副作用函数进行包装
276 | fetch(url)
277 | .then(JSON.parse)
278 | .then(filter(compose(equals('lucifer'), get('user'))))
279 | .then(chain(get('posts')))
280 | .then(map(get('title')))
281 | ```
282 |
283 | 看到了嘛,整个过程我们没有提到data。 我们不需要提到data。
284 | 代码精简了很多,逻辑纯粹了很多。
285 |
286 | > 我们再也不必为变量命名而苦恼了
287 |
288 | 上面用到了一个api`chain`, 可能不太好理解,我会在函数式编程-进阶部分讲解。
289 | 现在你可以把它理解为将数组拍平,就好像我在非函数式写法中的那样。
290 |
291 | 另一个比较难以理解的地方在于`filter(compose(equals('lucifer'), get('user')))`
292 |
293 | 我们来看一下:
294 |
295 | ```js
296 | // 这样的写法更容易理解,但是它不pointfree
297 | filter(data => equals(get('user')(data), 'lucifer'))
298 |
299 | // 等价于下面的写法(交换律)
300 |
301 | filter(data => equals('lucifer', get('user')(data)))
302 |
303 | // 等价于下面的写法(curry)
304 | filter(data => equals('lucifer')(get('user')(data)))
305 |
306 | // 把equal('lucifer') 看成f, get('user') 看成g
307 | // 上面的代码本质上是f(g(x))
308 | // 因此等价于下面的写法(compose) f(g(x)) = compose(f, g)(X)
309 | filter(data => compose(equals('lucifer'), get('user'))(data))
310 |
311 | // 所有的形如 x => fn(x) 的代码都等价于 fn
312 | // 因此上面的代码等价于
313 | filter(compose(equals('lucifer'), get('user')))
314 | ```
315 | 通过curry, compose,data goes last一系列技巧,我们写出了pointfree的
316 | 代码。 pointfree风格是函数式编程中特别重要的概念。pointfree使得开发者写出的
317 | 代码更加容易重用,仅仅面向逻辑,而将具体的数据抽离出来。而且直观上来讲,代码
318 | 更加简洁。这还仅仅是一个小小的例子,现实中情况会复杂地多,其重要性不言而喻。
319 |
320 | ## 总结
321 | 本文从数学中的函数入手,讲述了函数式编程中的函数其实就是数学中的函数。
322 | 接着我们讲述了纯函数以及其优点。然后我们讲述了函数式编程中的两个基础概念,curry和compose。
323 | 最后阐述了pointfree的概念,并通过curry,compose,以及data goes last原则写出了一个pointfree风格的代码。
324 |
325 |
--------------------------------------------------------------------------------
/examples/news/db/news.json:
--------------------------------------------------------------------------------
1 | {
2 | "items": [
3 | {
4 | "title": "hello world1",
5 | "author": "lucifer",
6 | "cover":
7 | "http://www.ruanyifeng.com/blogimg/asset/201204/bg2012040601.png",
8 | "content": "新消息",
9 | "publishDate": "lucifer"
10 | },
11 | {
12 | "title": "hello world2",
13 | "author": "lucifer",
14 | "cover":
15 | "",
16 | "content": "新消息",
17 | "publishDate": "lucifer"
18 | },
19 | {
20 | "title": "hello world3",
21 | "author": "lucifer",
22 | "cover":
23 | "",
24 | "content": "新消息",
25 | "publishDate": "lucifer"
26 | },
27 | {
28 | "title": "hello world4",
29 | "author": "lucifer",
30 | "cover":
31 | "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcQnJcMQAjjH--kvx_iPKxuNCwu0kA11xhJwDVY4eCMvxhEjqmOI",
32 | "content": "新消息",
33 | "publishDate": "lucifer"
34 | },
35 | {
36 | "title": "hello world5",
37 | "author": "lucifer",
38 | "cover":
39 | "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcTBLMiFaF6bhidw9cKB8ujJU7d8ZnenAITFNuayci9pChqndsPbqg",
40 | "content": "新消息",
41 | "publishDate": "lucifer"
42 | }
43 | ]
44 | }
45 |
--------------------------------------------------------------------------------
/examples/news/fp-example-news.xml:
--------------------------------------------------------------------------------
1 |