面试高频考点——手写compose

引言

在 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

我们很快就能想到答案,结果也完全没有问题。但这或许是许多初学者,或代码经验不丰富的程序员常见的问题。

但这有很多问题:几个函数之间耦合性太高,如果toUpperCasetoLowerCase或是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 函数接受两个函数 fg,并返回一个新的函数。新函数会先调用 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;
  }
}

总结

函数组合是函数式编程中的一个重要概念,它允许我们将多个函数组合成一个新的函数,从而实现更复杂的功能。在面试中,高阶函数、闭包、函数组合的实现及其应用场景是常见的考点。

相关推荐
hpoenixf4 小时前
2026 年前端面试问什么
前端·面试
还是大剑师兰特4 小时前
Vue3 中的 defineExpose 完全指南
前端·javascript·vue.js
泯泷4 小时前
阶段一:从 0 看懂 JSVMP 架构,先在脑子里搭出一台最小 JSVM
前端·javascript·架构
mengchanmian5 小时前
前端node常用配置
前端
华洛5 小时前
利好打工人,openclaw不是企业提效工具,而是个人助理
前端·javascript·产品经理
xkxnq5 小时前
第六阶段:Vue生态高级整合与优化(第93天)Element Plus进阶:自定义主题(变量覆盖)+ 全局配置与组件按需加载优化
前端·javascript·vue.js
RainyJiang6 小时前
谱写Kotlin协程面试进行曲-进阶篇(第二乐章)
面试·kotlin·android jetpack
A黄俊辉A6 小时前
vue css中 :global的使用
前端·javascript·vue.js
小码哥_常7 小时前
被EdgeToEdge适配折磨疯了,谁懂!
前端
小码哥_常7 小时前
从Groovy到KTS:Android Gradle脚本的华丽转身
前端