引言
在 JavaScript 中,函数是一等公民,这意味着函数可以作为参数传递,也可以作为返回值返回。这种特性使得 JavaScript 非常适合函数式编程(Functional Programming,FP)。函数式编程中的一个重要概念是函数组合(Function Composition),它允许我们将多个函数组合成一个新的函数,从而实现更复杂的功能。
在面试中,手写compose
函数是一个常考点,本文将深入探讨这个问题。
高阶函数
高阶函数(Higher-Order Function)是指那些接受一个或多个函数作为输入参数的函数,或者返回一个函数作为结果的函数。这两种情况都可以单独出现,也可以同时出现在一个高阶函数中。
高阶函数是函数式编程的核心概念之一,它使得我们可以将函数抽象出来,实现更灵活的代码设计。
函数组合
题目要求:将以下函数封装成一个函数,实现大写并拼接的功能
javascript
// 字符大写
var toUpperCase = function(x) {
return x.toUpperCase();
};
// 小写
var toLowerCase = function(x) {
return x.toLowerCase();
}
// 字符拼接
var hello = function(x) {
return 'HELLO, ' + x;
};
初步实现
js
var greet = function(x){
return hello(toUpperCase(x))
}
var greet2 = function(x){
return hello(toLowerCase(x))
}
console.log(greet('zhang'),greet('ZHANG')) // ZHANG zhang
我们很快就能想到答案,结果也完全没有问题。但这或许是许多初学者,或代码经验不丰富的程序员常见的问题。
但这有很多问题:几个函数之间耦合性太高,如果toUpperCase
、toLowerCase
或是hello
函数被修改添加新的功能之后,greet
的结果就可能发生错误,也要同时进行修改。我们针对这一问题,对代码进行完善。
进阶,降低耦合性
javascript
var compose = function(f, g) {
return function(x) {
return f(g(x));
};
};
var greet = compose(hello, toUpperCase);
console.log(greet('zhang')); // ZHANG
这个 compose
函数接受两个函数 f
和 g
,并返回一个新的函数。新函数会先调用 g(x)
,然后将结果传递给 f
。
这样,我们想要使用那个函数就传入哪个函数进行使用。并且降低了各个函数之间的依赖性。
但是,我们可以把函数变得更为完美一些,可以允许传入任意个参数。
最终的compose,支持多个函数的组合
手写 compose
函数是面试中的一个高频考点。你需要掌握以下几点:
- 如何接受多个函数作为参数(es6 剩余参数... 或是 arguments)。
- 如何从右到左依次调用这些函数(args.length - 1)。
- 如何使用闭包保存参数列表(外部函数返回了一个内部函数,该内部函数能够访问外部函数的变量)。
javascript
var compose = function(...args) { // es6 剩余参数
// var args = arguments; // 类数组 拿到所有参数
var start = args.length - 1; // 最右边的函数
return function(x) {
var i = start;
var result = args[start].call(this, x); // 本例中只有一个参数,就先使用call了
while(i--) {
result = args[i].call(this, result);
}
return result;
}
}
总结
函数组合是函数式编程中的一个重要概念,它允许我们将多个函数组合成一个新的函数,从而实现更复杂的功能。在面试中,高阶函数、闭包、函数组合的实现及其应用场景是常见的考点。