函数声明会被提升,可提前调用;函数表达式不会提升,须定义后调用;箭头函数无this、arguments,不可构造;块级函数声明在严格模式下非法;应据场景选型并重视函数命名。
JavaScript 中 function 关键字开头的语句,如果直接出现在脚本顶层或函数体顶层(不被赋值、不被括号包裹),就是**函数声明**;而如果它作为表达式的一部分出现(比如赋值给变量、传入参数、放在括号里),就是**函数表达式**。最关键的区别在于:函数声明会被**提升(hoisting)**,函数表达式不会。
function foo() { return 'declared'; } —— 可以在定义前调用const bar = function() { return 'expressed'; }; —— 必须在赋值后才能调用,否则报 ReferenceError 或 TypeError
const fn = function() {})在调试器中显示为 fn: ƒ anonymous(),可读性差;建议用命名函数表达式:const fn = function namedFn() {},此时 namedFn 在函数体内可用,且 DevTools 能正确显示名称虽然箭头函数常被赋值给变量(const fn = () => {}),但它本身是表达式,且**没有自己的 this、arguments、super 或 new.target**。它不适用于需要动态 this 的场景(比如事件处理器中想访问触发元素,或对象方法中想访问实例)。
new (() => {}) 报 TypeError: xxx is not a constructor
arguments 对象,需改用剩余参数:const sum = (...nums) => nums.reduce((a, b) => a + b)
const square = x => x * x,但一旦加逻辑(如条件判断),必须加花括号和 return
早期非严格模
式下,浏览器允许在 if 块里写 function foo() {},但行为不统一(有的提升到函数作用域,有的只在块内可见)。ES6 明确规定:**函数声明在块级作用域中属于“非法语法”(early error)**,严格模式下直接抛 SyntaxError。
if (true) {
function f() { return 1; }
}
// 严格模式下:Uncaught SyntaxError: Function declarations not allowed in single-statement context若需条件定义函数,请用函数表达式:
立即学习“Java免费学习笔记(深入)”;
let f;
if (true) {
f = function() { return 1; };
}函数声明适合模块顶层的、明确的、会被多次调用的工具函数(如 function validateEmail(str) {...}),利于阅读和调试;函数表达式更适合回调、IIFE、或需要闭包捕获变量的场景;箭头函数则优先用于简短的、无状态的转换逻辑(如 array.map(x => x.id))。
const fn = function foo() {} 还同时依赖 foo 名称做递归——应统一用 function foo() { return foo(); }(声明)或显式引用变量(const fn = function() { return fn(); };)export function foo() {},语义清晰且支持静态分析函数名是否出现在调用栈、是否能被压缩工具安全重命名、是否参与作用域提升——这些细节在重构或排查线上问题时,往往比语法糖更影响效率。