面试高频考点——手写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;
  }
}

总结

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

相关推荐
庸俗今天不摸鱼1 分钟前
【万字总结】前端全方位性能优化指南(十)——自适应优化系统、遗传算法调参、Service Worker智能降级方案
前端·性能优化·webassembly
QTX187302 分钟前
JavaScript 中的原型链与继承
开发语言·javascript·原型模式
黄毛火烧雪下8 分钟前
React Context API 用于在组件树中共享全局状态
前端·javascript·react.js
Apifox19 分钟前
如何在 Apifox 中通过 CLI 运行包含云端数据库连接配置的测试场景
前端·后端·程序员
一张假钞22 分钟前
Firefox默认在新标签页打开收藏栏链接
前端·firefox
高达可以过山车不行22 分钟前
Firefox账号同步书签不一致(火狐浏览器书签同步不一致)
前端·firefox
m0_5937581023 分钟前
firefox 136.0.4版本离线安装MarkDown插件
前端·firefox
掘金一周26 分钟前
金石焕新程 >> 瓜分万元现金大奖征文活动即将回归 | 掘金一周 4.3
前端·人工智能·后端
三翼鸟数字化技术团队44 分钟前
Vue自定义指令最佳实践教程
前端·vue.js
uhakadotcom1 小时前
构建高效自动翻译工作流:技术与实践
后端·面试·github