成为一名函数式码农(2)

上一篇成为一名函数式码农(1)

友情提示

请仔细阅读文中的代码。确保你已经理解了代码之后再进行下一步。每一节都是建立在前一节的基础上。

如果你匆忙行事,你可能会错过一些对后面章节很重要的细微差别。

重构

我们先来看一下重构问题。这里有一些JavaScript代码片段:

<pre>
function validateSsn(ssn) {
if (/^\d{3}-\d{2}-\d{4}$/.exec(ssn))
console.log('Valid SSN');
else
console.log('Invalid SSN');
}
function validatePhone(phone) {
if (/^(\d{3})\d{3}-\d{4}$/.exec(phone))
console.log('Valid Phone Number');
else
console.log('Invalid Phone Number');
}
</pre>

我们先前都这样写代码,一段时间后我们开始意识到这两个函数实际上功能是一样的只有一小部分不一样(已经以粗体显示)。

我们不应该复制粘贴validateSsn然后修改一些代码来创建validatePhone,而应该创建一个单独的函数通过参数来处理粘贴以后修改的部分。

在本例中我们应该通过参数处理valueregular expression和打印的message(至少打印消息的最后一部分)。

重构后的代码:

<pre>
function validateValue(value, regex, type) {
if (regex.exec(value))
console.log('Invalid ' + type);
else
console.log('Valid ' + type);
}
</pre>

value代表之前代码中的ssnphone

regex代表之前的正则表达式/^\d{3}-\d{2}-\d{4}$//^\(\d{3}\)\d{3}-\d{4}$/

最后,消息的最后一部分即‘SSN’‘Phone Number’type表示。

使用一个函数比使用两个函数或者更糟的3个、4个或者10个函数要好一些。这使得代码整洁并且可维护。

例如,如果发现一个bug,你只需要修改一个地方即可,而不是搜索整个代码库来查找可能粘贴并修改这个函数的地方。

但是如果有以下这样的情况会怎样:

<pre>
function validateAddress(address) {
if (parseAddress(address))
console.log('Valid Address');
else
console.log('Invalid Address');
}
function validateName(name) {
if (parseFullName(name))
console.log('Valid Name');
else
console.log('Invalid Name');
}
</pre>

在这parseAddressparseFullName函数都是接受一个string,如果传入的字符串符合预订的句法则返回true

我们怎么来重构这段代码?

好的,我们可以像前面一样使用value来代表addressname,使用type来代表‘Address’‘Name’,但是我们的正则表达式之前是一个函数。

如果我们可以将一个函数作为参数传递。。。

高阶函数

许多语言不支持将函数作为参数传递。有的支持但是变得非常复杂。

在函数式语言中,函数是这个语言中的一等公民。换句话说,函数仅仅是另一种值而已。

由于函数仅仅是值,我们可以将它们作为参数传递。

尽管JavaScript不是纯粹的函数式语言,你可以用它做一些函数式操作。所以这里我们将前面两个函数重构成一个函数,将parsing function(正则表达式匹配函数)通过参数parseFunc传递进去:

<pre>
function validateValueWithFunc(value, parseFunc, type) {
if (parseFunc(value))
console.log('Invalid ' + type);
else
console.log('Valid ' + type);
}
</pre>

我们的新函数就称为高阶函数(Higher-order Function)

高阶函数可以接收函数作为参数,或者返回一个函数结果,或者两者同时具备。

现在可以使用我们的高阶函数来替代前的4个函数(这个在JavaScript中可以工作,因为Regex.exec匹配成功会返回true):

validateValueWithFunc('123-45-6789', /^\d{3}-\d{2}-\d{4}$/.exec, 'SSN');
validateValueWithFunc('(123)456-7890', /^\(\d{3}\)\d{3}-\d{4}$/.exec, 'Phone');
validateValueWithFunc('123 Main St.', parseAddress, 'Address');
validateValueWithFunc('Joe Mama', parseName, 'Name');

这比前面使用4个几乎一模一样的函数要好多了。

但是请注意正则表达式。它们有一点啰嗦。我们再重构一下:

<pre>
var parseSsn = /^\d{3}-\d{2}-\d{4}$/.exec;
var parsePhone = /^(\d{3})\d{3}-\d{4}$/.exec;
validateValueWithFunc('123-45-6789', parseSsn, 'SSN');
validateValueWithFunc('(123)456-7890', parsePhone, 'Phone');
validateValueWithFunc('123 Main St.', parseAddress, 'Address');
validateValueWithFunc('Joe Mama', parseName, 'Name');
</pre>

好一点了。现在当我们想要分析一个电话号码时,我们不需要再复制粘贴正则表达式。

但是想象一下我们有更多的正则表达式要分析,不仅仅是parseSsnparsePhone。每次我们创建一个正则表达式解析器,我们需要记住添加.exec到它的末尾。说真的,对我来说很容易忘记。

我们可以通过创建一个返回exec函数的高阶函数来防止这种情况:

<pre>
function makeRegexParser(regex) {
return regex.exec;
}
var parseSsn = makeRegexParser(/^\d{3}-\d{2}-\d{4}$/);
var parsePhone = makeRegexParser(/^(\d{3})\d{3}-\d{4}$/);
validateValueWithFunc('123-45-6789', parseSsn, 'SSN');
validateValueWithFunc('(123)456-7890', parsePhone, 'Phone');
validateValueWithFunc('123 Main St.', parseAddress, 'Address');
validateValueWithFunc('Joe Mama', parseName, 'Name');
</pre>

这里makeRegexParser接收一个正则表达式并返回exec函数,exec函数接收一个字符串参数。validateValueWithFunc函数会传递字符串(value)给解析函数,即exec

的确,这是一个微不足道的改进,但是这里展示了一个高阶函数返回一个函数的例子。

但是,你可以想象一下如果makeRegexParser比现在要复杂得多的时候,这个改进所带来的好处。

这里还有另外一个高阶函数返回函数的例子:

function makeAdder(constantValue) {
    return function adder(value) {
        return constantValue + value;
    };
}

makeAdder接收一个参数constantValue返回一个adder,adder是一个函数,它将给它接收的参数加上一个常量值。

这是它怎么被使用:

var add10 = makeAdder(10);
console.log(add10(20)); // prints 30
console.log(add10(30)); // prints 40
console.log(add10(40)); // prints 50

我们通过向makeAdder函数(它会返回一个函数)传递一个常量10来创建一个函数add10add10将给任意值加上10

注意adder函数可以访问constantValue,即使makeAddr函数已经返回。因为adder在创建时constantValue在它的作用域内。

这个行为非常的重要,因为如果没有它,能够返回函数的函数不是很有用。所以我们理解它是怎么工作以及怎么称呼它非常要。

这种行为成为闭包

闭包(Closures)

这里有一个人为的使用闭包的函数例子:

<pre>
function grandParent(g1, g2) {
var g3 = 3;
return function parent(p1, p2) {
var p3 = 33;
return function child(c1, c2) {
var c3 = 333;
return g1 + g2 + g3 + p1 + p2 + p3 + c1 + c2 + c3;
};
};
}
</pre>

在这个例子中,child可以访问它自己的变量,parent的变量以及grandParent的变量。

parent可以访问它自己的变量和grandParent的变量。

grandParent只能访问它自己的变量。

(可以参考上面的金字塔来理解)

这里是它的使用范例:

var parentFunc = grandParent(1, 2); // returns parent()
var childFunc = parentFunc(11, 22); // returns child()
console.log(childFunc(111, 222)); // prints 738
// 1 + 2 + 3 + 11 + 22 + 33 + 111 + 222 + 333 == 738

这里,parentFunc保持parent的作用域存活,因为grandParent返回的是parent函数。

类似的,childFunc保持child的作用域存活,因为parentFunc(这里就是parent)返回child函数。

当一个函数被创建,其整个生命周期中都是可以访问在在其创建时作用域内的所有变量。只要有引用指向它该函数就会一直存在。例如:只要childFunc还引用它,child的作用域就一直存在。

闭包就是一个函数的作用域,这个作用域通过指向该函数的引用保持存活。

注意在JavaScript中闭包是会造成问题的,因为这些变量是可修改的,即,从它们结束一直到他们返回的函数被调用期间,它们都可以修改变量。

幸亏,函数式语言中的变量都是不可修改的,这样可以消除这个常见的bug以及困惑根源。

下一篇:成为一名函数式码农(3)

本文译自So You Want to be a Functional Programmer (Part 2)

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

推荐阅读更多精彩内容