高阶函数是接受函数作为参数或返回函数作为结果的函数。它体现JavaScript原生支持的函数式编程能力,如map、filter、reduce均通过接收回调函数实现解耦;once等自定义高阶函数利用闭包封装状态;使用时需注意this绑定与作用域问题。
高阶函数不是“高级”的函数,而是指**接受函数作为参数**或**返回函数作为结果**的函数。这是 JavaScript 中函数式编程的基础能力,不是语法糖,而是语言原生支持的核心特性。
map、filter、reduce 都是高阶函数它们都接收一个回调函数作为第一个参数,并在其内部调用它。这不是约定,是设计使然:
map 把每个元素传给回调,收集返回值生成新数组filter 用回调的布尔返回值决定是否保留当前元素reduce 把回调当作累加器逻辑,每次迭代调用它这些方法本身不关心回调具体做什么——它只负责“在合适时机、以合适方式调用你给的函数”。这种解耦正是高阶函数的价值。
once 防重复执行常见
需求:某个函数只能运行一次,后续调用直接返回上次结果。这需要闭包 + 状态标记:
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)背后就是高阶函数在起作用。