高阶函数是接受函数作为参数或返回函数作为结果的函数。它体现JavaScript原生支持的函数式编程能力,如map、filter、reduce均通过接收回调函数实现解耦;once等自定义高阶函数利用闭包封装状态;使用时需注意this绑定与作用域问题。

高阶函数不是“高级”的函数,而是指**接受函数作为参数**或**返回函数作为结果**的函数。这是 JavaScript 中函数式编程的基础能力,不是语法糖,而是语言原生支持的核心特性。
为什么 map、filter、reduce 都是高阶函数
它们都接收一个回调函数作为第一个参数,并在其内部调用它。这不是约定,是设计使然:
-
map把每个元素传给回调,收集返回值生成新数组 -
filter用回调的布尔返回值决定是否保留当前元素 -
reduce把回调当作累加器逻辑,每次迭代调用它
这些方法本身不关心回调具体做什么——它只负责“在合适时机、以合适方式调用你给的函数”。这种解耦正是高阶函数的价值。
自己写一个高阶函数:比如 once 防重复执行
常见需求:某个函数只能运行一次,后续调用直接返回上次结果。这需要闭包 + 状态标记:
立即学习“Java免费学习笔记(深入)”;
function once(fn) {
let called = false;
let result;
return function(...args) {
if (called) return result;
result = fn.apply(this, args);
called = true;
return result;
};
}
使用时:
const safeInit = once(() => console.log('init!'));- 第一次调用
safeInit()打印 init!,第二次起静默返回undefined - 注意:
once返回的是新函数,原函数fn并未被修改
容易踩的坑:忘记绑定 this 或丢失参数
当把方法(如对象上的函数)传给高阶函数时,this 会丢失:
- 错误写法:
arr.map(obj.method)→method内部的this指向undefined(严格模式) - 正确写法之一:
arr.map(x => obj.method(x)) - 或提前绑定:
arr.map(obj.method.bind(obj)) - 再比如:
setTimeout(callback, 1000)中,如果callback是箭头函数且依赖外部变量,要注意作用域是否符合预期
高阶函数本身不难写,难的是判断何时该封装行为、何时该暴露控制权。多数时候,真正卡住人的不是“怎么写”,而是“要不要在这里抽象出一个函数参数”——这取决于逻辑是否多变、是否需要复用、是否涉及异步或副作用。别为了高阶而高阶,但要清楚哪些 API(比如 fetch 的拦截器、React 的 useCallback)背后就是高阶函数在起作用。











