高阶函数是接受函数作为参数或返回函数作为结果的函数。典型内置高阶函数包括map、filter、reduce、sort、find、some、every和forEach;自定义时只需满足任一条件即可;常见错误是误调用而非传引用;性能影响极小,关键在回调逻辑本身。

高阶函数不是“高级的函数”,而是指**接受函数作为参数,或返回函数作为结果**的函数。这是 JavaScript 中函数式编程的基础能力,不是语法糖,而是语言原生支持的核心特性。
哪些内置方法是典型的高阶函数?
JavaScript 数组原型上多个方法都属于高阶函数,因为它们都要求传入一个 callback 函数:
-
map():对每个元素调用callback,返回新数组 -
filter():用callback判断是否保留当前元素 -
reduce():用callback累积计算,返回单个值 -
sort()(可选):传入比较函数(a, b) => number控制排序逻辑 -
find()、some()、every():全部依赖返回布尔值的callback
注意:forEach() 也是高阶函数,但它不返回新数组(返回 undefined),所以不能链式调用 —— 这是它和 map() 的关键区别。
自己写的函数怎么变成高阶函数?
只要满足“接收函数”或“返回函数”任一条件,就是高阶函数。常见写法有:
立即学习“Java免费学习笔记(深入)”;
- 接收函数参数:
function throttle(fn, delay) { return function() { ... }; } - 返回函数(闭包):
const createLogger = (prefix) => (msg) => console.log(`[${prefix}] ${msg}`); - 组合函数:
const compose = (f, g) => (x) => f(g(x));
这类函数本身不直接处理数据,而是“配置行为”或“封装调用时机”,比如防抖、日志增强、权限校验中间件等场景都依赖这种模式。
为什么传函数进去却报错“xxx is not a function”?
这是最常踩的坑:你传进去了,但没传对。典型原因包括:
- 把函数调用写成函数引用:
arr.map(callback())❌ 应该是arr.map(callback)✅ - 箭头函数少写了括号导致隐式返回对象字面量:
arr.map(x => { value: x })❌ 实际返回undefined;应写成arr.map(x => ({ value: x }))✅ - 上下文丢失:在对象方法中直接传
obj.handler给addEventListener,this会变;需用obj.handler.bind(obj)或() => obj.handler()
错误信息里出现 is not a function,第一反应不该是“函数不存在”,而应检查它是不是被意外执行了、或者被覆盖成了其他类型值。
高阶函数和普通函数调用性能差多少?
几乎没有运行时开销。V8 引擎对闭包和函数参数早做了深度优化。真正影响性能的是回调函数内部的逻辑(比如在 map() 里做大量 DOM 操作),而不是“多套了一层函数调用”。但要注意:
- 频繁创建匿名函数(如在循环里写
() => x * 2)会增加 GC 压力,可提取为具名常量复用 -
bind()返回的新函数每次都会新建,比箭头函数或直接传更重;优先用class方法或useCallback(React)缓存 - 嵌套过深的高阶函数(如五层
compose(a)(b)(c)(d)(e))会让调试栈变长,不利于排查
实际项目里,别为了“函数式”而堆叠高阶函数;清晰表达意图比形式更重要 —— 比如用 filter().map() 就比手写一个 filterMap() 更易懂。










