初入函数式编程

真·函数式编程 学习函数式编程的学习笔记

  • 禁用var/let,所有东西都用const定义,也就是说无变量,强制immutable。
  • 禁用分号,也就是不让“顺序执行”,解除过程式编程范式
  • 禁用if/else,但允许用条件表达式condition ? expr1 : expr2。
  • 禁用for/while/do-while。
  • 禁用prototype和this来解除JS中的面向对象编程范式
  • 禁用function和return关键字,仅用lambda表达式来编程(JS里叫箭头函数)。
  • 禁用多参数函数,只允许使用单个参数,相当于强行curry化

根据作者的提示, 先自宫一刀!
啊啊啊~ 好疼~
不过蛮爽的, 嘿嘿嘿!

ES 基本的箭头函数啥的略过

实现循环

命令式:

for (var i = 0; i < arr.length; i++) {
    // ...
}

这个太常见了, 也太简单了,第一天学编程就是这个玩意儿吧.

题外话: 关于这个例子 var i = 0 部分在 javascript 这个 var 写了和没写没啥区别.
使用 ES2015 改成 let 吧!

好! 来直接看 函数式 怎么写:

const two_steps = step1 => step2 => param =>
  step2(step1(param))

const loop_on_array = arr => body => i =>
    i < arr.length
    ? two_steps(body)(_ => loop_on_array(arr)(body)(i + 1))(arr[i])
    : undefined

别灰心,我第一次看也看不懂.

看个过度版本的例子:

function loop_on_array(arr, body, i) {
    if (i < arr.length) {
        body(arr[i])
        loop_on_array(arr, body, i + 1)
    }
}

这个都能看懂, 就是回调自己, 递归嘛.

但是根据 之前的 禁令, 貌似写了的都违背了.

回到之前的函数式写法, 再复制一遍

const two_steps = step1 => step2 => param =>
  step2(step1(param))

const loop_on_array = arr => body => i =>
  i < arr.length
  ? two_steps (body)(_ => loop_on_array(arr)(body)(i + 1))(arr[i])
  : undefined

现在是不是稍微能理解一点了,
就是用三目运算符代替 if/else
写了个 two_step 用作递归.

嗯,没错 two_step 就是用来递归的, 看懂这里你就可以 × 掉本页面了!

再来看下如何使用这个函数式的循环

loop_on_array([1, 2, 3, 4, 5])(item => console.log(item))(0)

对的, 没错我们写了个蛋疼的 0 (暂时先别管这个)

咱们遍历这个数组的时候, 第一个curry参数放的数组, 第二个放的一个函数, 第三个是个蛋疼 0
好大家简单做个映射, 等下要用的.

arr  = [1,2,3,4,5]
body = [item => console.log]
i    = 0

啥是curry化? 简而言之就是多个参数变成一个参数的写法, 相关的lambda演算.

上面参数的顺序可以通过调整对应 curry 参数位置调整的, 因为函数式不是按顺序执行的

咱反过来推倒, 这个例子的效果其实很简单, 就是打印数组里的所有内容.

  • 0 必然是作为第一次使用的下标值
  • item => console.log(item) 接受的参数 item 也肯定是[1,2,3,4,5]里的单个元素
  • 怎么样使它回调自己呢? 那就是使用 two_steps 方法了

好看看在 loop_on_array 中是如何使用 two_steps

回想下之前的映射关系.

two_steps (body)(_ => loop_on_array(arr)(body)(i + 1))(arr[i])

那么上式中:

arr[i] = arr[0] = 1
body   = item => console.log(item)

啥都不管, 我貌似看到可以先打印出一个 1, 只要吧 arr[0] 当作变量传给 body 就可以了
打印完第一个, 我肯定要接着打第二个,
这么看来使用 loop_on_array([1, 2, 3, 4, 5])(item => console.log(item))(0)
就打印了下标为 0arr 数组
那么 使用 loop_on_array([1, 2, 3, 4, 5])(item => console.log(item))(1)
就可以打印下标为 1arr 数组了

(_ => loop_on_array(arr)(body)(i + 1)) 这里不就是这么用的吗.

等等, 在 loop_on_array 的三目表达式中 只要 i < arr.length 就会执行 two_steps
也就是说 这会递归调用自己.

这时候看 two_steps 就能看懂了吧.

const two_steps = step1 => step2 => param => step2(step1(param))

two_steps 接受3个 curry化参数:

  1. step1 也就是第一次要执行的方法, 例子中的 item => console.log(item)
  2. step2 也就是递归调用的, _ => loop_on_array(arr)(body)(i + 1)的写法也能看出他根本不在乎传入的参数, 他只是用来回调 loop_on_array的 所以使用了_来忽略传入参数
  3. paramsstep1 中是当前的下标, +1 之后就会是step2parmas
// 这里的3个参数顺序调换对应的只要吧 `step2(step1(param))` 就可以了
// 比如我写成 `const two_steps = step1 => param => step2 => step2(step1(param))`
? two_steps(body)(_ => loop_on_array(arr)(body)(i + 1))(arr[i]) 
// 改成
? two_steps(body)(arr[i])(_ => loop_on_array(arr)(body)(i + 1))
// (body)(arr[i])这样放近一点我更容易理解 arr[i] 是作为 body 的参数
// 然后再回调 (_ => loop_on_array(arr)(body)(i + 1)) 这个方法

刚了解函数式可能觉得, OMG 写个破循环这么费劲.
是的, 我也是这么认为的...
那是因为咱们还没见识到她的魅力~

关于这个蛋疼的0其实很好解决:

const new_loop = arr => body => loop_on_array(arr)(body)(0)

能看出一点他的魅力吗?

前端正在革命, 我们不要做切图仔!

此文写给不愿意做切图仔的你我.
参考: http://web.jobbole.com/84882/

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352

推荐阅读更多精彩内容