聊一聊JavaScript中的立即执行函数(IIFE)

一、为什么需要IIFE?

假设咱在代码里定义了一个变量:

javascript 复制代码
var a = 2;
function foo() {
    var a = 3;
    console.log("内部a:", a); // 3
}
foo();
console.log("外部a:", a); // 2

看起来没问题对吧?但用函数声明可能会不小心污染全局作用域。于是有人想到:能不能让函数定义完立刻执行,还不用起名字?

这时候IIFE就登场了


二、IIFE的基本套路

核心原理:把函数变成表达式

javascript 复制代码
var a = 2;
// 注意这两个小括号!
(function() {
    var a = 3;
    console.log("偷偷改a:", a); // 3
})(); // 第一个括号把函数变成表达式,第二个括号立刻执行它

console.log("外面的a还是:", a); // 2

关键点:

  • ()包裹函数,让它从"函数声明"变成"函数表达式"
  • 末尾加()立刻调用,不用手动执行
  • 内部变量和外部完全隔离,互不干扰

三、IIFE的进阶玩法

1. 参数传递:把全局对象当快递寄进去

javascript 复制代码
var a = 2;
(function(global) { // 这里收快递
    var a = 3;
    console.log("内部a:", a); // 3
    console.log("外面的a其实在global里:", global.a); // 2
})(window); // 把window当参数传进去

// 好处:不用直接写window,代码更灵活

2. 防坑指南:保护undefined

有时候手贱会干这种事:

javascript 复制代码
undefined = "嘿嘿,我篡改了!"; // 给其他代码挖了个大坑
(function(undefined) { // 这里设个陷阱
    var a;
    if (a === undefined) {
        console.log("这里的undefined很安全!"); // 正常输出
    }
})(); // 不传参数,undefined参数自动变成真正的undefined

原理: 不传参数时,函数内的undefined参数值就是原始undefined,外面的骚操作影响不到里面!

将被undefined标识符的默认值被错误覆盖导致的异常的代码放入即可验证和解决该问题。

3. 顺序控制:把代码倒着写

javascript 复制代码
var a = 2;
// 先定义执行逻辑
(function(def) {
    def(window); // 第二步:执行传进来的函数
})(function(global) { // 第一步:把函数当参数传进去
    var a = 3;
    console.log("内部a:", a); // 3
    console.log("全局a:", global.a); // 2
});

整体逻辑: 函数表达式def定义在片段的第二部分,然后当做参数(这个参数也叫做def)被传递进IIFE函数定义的第一部分,最后参数def(也就是传进去的函数)被调用,并将window传入当做global的参数的值。

适用场景: 比如某个函数需要先准备好才能执行,可以用这种"传参倒序"的方式组织代码。


四、IIFE的日常妙用

场景1:快速创建一个独立沙箱

javascript 复制代码
// 临时需要一个干净的作用域
(function() {
    var tempData = "敏感数据";
    // 这里随便折腾,不会影响外面
})();
// tempData在这里访问不到,自动销毁

场景2:保护第三方库变量

很多库(比如老版本jQuery)会这么写:

javascript 复制代码
(function(window) {
    var privateKey = "保密!";
    window.$ = function() { /* ... */ };
})(window);
// 外面只能访问到$,privateKey被藏起来了

总结:IIFE到底解决了啥?

  • 隔离作用域:关起门来操作变量,不怕和外面冲突
  • 灵活传参:把全局对象、依赖项当参数传递,代码更可控
  • 防止污染:避免不小心修改undefined等特殊值

虽然现在ES6的let/const和模块化逐渐替代了IIFE,但理解它依然能帮你看懂很多老代码的设计思路。下次见到这种(function(){ ... })()结构,你就能会心一笑啦!

相关推荐
zhougl9961 小时前
html处理Base文件流
linux·前端·html
花花鱼1 小时前
node-modules-inspector 可视化node_modules
前端·javascript·vue.js
HBR666_1 小时前
marked库(高效将 Markdown 转换为 HTML 的利器)
前端·markdown
careybobo3 小时前
海康摄像头通过Web插件进行预览播放和控制
前端
TDengine (老段)3 小时前
TDengine 中的关联查询
大数据·javascript·网络·物联网·时序数据库·tdengine·iotdb
杉之4 小时前
常见前端GET请求以及对应的Spring后端接收接口写法
java·前端·后端·spring·vue
喝拿铁写前端4 小时前
字段聚类,到底有什么用?——从系统混乱到结构认知的第一步
前端
再学一点就睡4 小时前
大文件上传之切片上传以及开发全流程之前端篇
前端·javascript
木木黄木木5 小时前
html5炫酷图片悬停效果实现详解
前端·html·html5
请来次降维打击!!!6 小时前
优选算法系列(5.位运算)
java·前端·c++·算法