惰性函数定义模式 使用方法第13页

2018-10-15 17:03

阅读:478

这篇文章阐述的是一种函数式编程(functional-programming)设计模式,我称之为惰性函数定义(LazyFunctionDefinition)。我不止一次发现这种模式在JavaScript中大有用处,尤其是编写跨浏览器的、高效运行的库之时。

热身问题

编写一个函数foo,它返回的是Date对象,这个对象保存的是foo首次调用的时间。

方法一:上古时代的技术

这个最简陋的解决方案使用了全局变量t来保存Date对象。foo首次调用时会把时间保存到t中。接下来的再次调用,foo只会返回保存在t中的值。
复制代码 代码如下:
vart;
functionfoo(){
if(t){
returnt;
}
t=newDate();
returnt;
}

但是这样的代码有两个问题。第一,变量t是一个多余的全局变量,并且在foo调用的间隔期间有可能被更改。第二,在调用时这些代码的效率并没有得到优化因为每次调用foo都必须去求值条件。虽然在这个例子中,求值条件并不显得低效,但在现实世界的实践例子中常常会有极为昂贵的条件求值,比如在if-else-else-…的结构中。

方法二:模块模式

我们可以通过被认为归功于Cornford和Crockford的模块模式来弥补第一种方法的缺陷。使用闭包可以隐藏全局变量t,只有在foo内的代码才可以访问它。

复制代码 代码如下:
varfoo=(function(){
vart;
returnfunction(){
if(t){
returnt;
}
t=newDate();
returnt;
}
})();

但这仍然没有优化调用时的效率,因为每次调用foo依然需要求值条件。
虽然模块模式是一个强大的工具,但我坚信在这种情形下它用错了地方。
方法三:函数作为对象
由于JavaScript的函数也是对象,所以它可以带有属性,我们可以据此实现一种跟模块模式质量差不多的解决方案。

复制代码 代码如下:
functionfoo(){
if(foo.t){
returnfoo.t;
}
foo.t=newDate();
returnfoo.t;
}

在一些情形中,带有属性的函数对象可以产生比较清晰的解决方案。我认为,这个方法在理念上要比模式模块方法更为简单。

这个解决方案避免了第一种方法中的全局变量t,但仍然解决不了foo每次调用所带来的条件求值。
123下一页阅读全文


评论


亲,登录后才可以留言!