0

0

深入理解JavaScript中的立即调用类表达式(IICE)及其执行机制

DDD

DDD

发布时间:2025-11-23 12:33:01

|

887人浏览过

|

来源于php中文网

原创

深入理解JavaScript中的立即调用类表达式(IICE)及其执行机制

本文深入探讨javascript中立即调用类表达式(iice)的执行机制,通过一个嵌套继承的示例,详细解析其声明、继承、实例化及构造函数调用顺序。我们将识别父子类关系,揭示属性初始化时序对结果的影响,并提供修正方案,旨在帮助开发者理解此类复杂结构的行为和潜在陷阱。

1. 立即调用类表达式(IICE)概览

在JavaScript中,立即调用函数表达式(IIFE)是一种常见的模式,用于创建私有作用域并立即执行。与此类似,立即调用类表达式(Immediately Invoked Class Expression, IICE)则是一种声明一个匿名类并立即实例化该类的模式。其基本形式为 new (class MyClass { /* ... */ })()。这种结构允许我们定义一个类,然后无需将其绑定到具名变量,直接创建一个实例。

考虑以下示例代码,它展示了一个嵌套的立即调用类表达式:

new (class C extends class B {
  constructor() {
    console.log(this.foo());
  }
} {
  num = 1;
  foo() {
    return this.num;
  }
})();

这段代码看似复杂,但通过逐步分解,我们可以清晰地理解其执行机制。

2. 代码结构解析与执行流程

为了更好地理解上述代码,我们将其从内到外进行分解。

立即学习Java免费学习笔记(深入)”;

2.1 内部类声明:父类 B

首先,我们关注最内部的类声明:

class B {
  constructor() {
    console.log(this.foo());
  }
}

这是一个标准的类声明,定义了一个名为 B 的类。在JavaScript中,类声明的本质是创建一个构造函数。因此,class B { ... } 实际上生成了一个名为 B 的构造函数。这个构造函数在实例化时会执行 console.log(this.foo())。

2.2 外部类声明:子类 C

接下来,我们将内部的 class B 视为一个已定义的实体(即构造函数 B),那么整个表达式可以简化为:

new (class C extends B {
  num = 1;
  foo() {
    return this.num;
  }
})();

这里,我们声明了另一个匿名类,为了便于理解,我们称之为 C。这个类 C 通过 extends B 继承了类 B。因此,B 是父类,C 是子类

类 C 包含一个实例属性 num,初始化为 1,以及一个方法 foo(),用于返回 this.num。

2.3 类的实例化

最后,整个表达式 new (C)() 或者简化为 new C(),表示立即实例化这个匿名子类 C。

Removal.AI
Removal.AI

AI移出图片背景工具

下载

2.4 构造函数执行顺序与方法调用

当 new C() 被执行时,其内部的执行流程如下:

  1. 创建实例对象: JavaScript引擎首先创建一个新的空对象,并将其原型链指向 C.prototype。
  2. 调用父类构造函数: 接着,会自动调用父类 B 的构造函数。在 B 的 constructor 中,执行 console.log(this.foo())。
    • 此时,this 指向的是正在被创建的 C 的实例。
    • this.foo() 会被调用。由于 C 继承了 B,并且 C 自身定义了 foo() 方法,因此会调用 C 实例上的 foo() 方法。
  3. 初始化子类实例属性: 父类构造函数执行完毕后,才会初始化子类 C 中声明的实例属性 num = 1。
  4. 调用子类构造函数: 如果子类 C 有自己的 constructor(本例中没有显式定义,但会有一个隐式的 constructor),则会执行。

3. 深入分析:属性初始化与执行时序问题

根据上述执行顺序,我们可以发现一个关键问题:在父类 B 的 constructor 中调用 this.foo() 时,子类 C 的实例属性 num = 1 尚未被初始化。

在JavaScript类的继承中:

  • 父类的构造函数会首先执行。
  • 子类的实例属性(如 num = 1)是在父类构造函数执行完毕后,但在子类自身构造函数(如果存在且显式调用 super())执行之前进行初始化的。

因此,当 B 的 constructor 调用 this.foo() 时,this.num 实际上还未被赋值,其值为 undefined。所以,原始代码的输出结果是 undefined。

// 原始代码执行结果
undefined

4. 解决方案与最佳实践

要解决 undefined 的问题,我们需要确保在 B 的 constructor 调用 this.foo() 时,num 属性已经存在并被正确初始化。一种直接的方法是将 num 属性的声明移动到父类 B 中。

new (class C extends class B {
  num = 1; // 将 num 属性移动到父类 B 中
  constructor() {
    console.log(this.foo());
  }
} {
  // num = 1; // 此处不再需要
  foo() {
    return this.num;
  }
})();

在这个修正后的版本中:

  1. 当 new C() 实例化时,会先创建实例对象。
  2. 调用父类 B 的 constructor。
  3. 在 B 的 constructor 执行之前,B 中声明的实例属性 num = 1 会被初始化到实例上。
  4. 然后 B 的 constructor 执行 console.log(this.foo())。此时 this.num 已经为 1。
  5. this.foo() 返回 1。

因此,修正后的代码将输出 1。

// 修正后代码执行结果
1

注意事项:

  • 属性初始化时机: 在涉及类继承和构造函数调用的复杂场景中,务必清晰理解实例属性的初始化时机。ES6 类中的实例属性(Class Field)在父类构造函数执行后、子类构造函数执行前初始化。
  • 设计考量: 立即调用类表达式通常用于创建单例、封装私有逻辑或动态生成类等高级场景。在设计此类结构时,应仔细考虑其生命周期、继承关系和属性访问权限。
  • 避免副作用: 尽量避免在父类构造函数中调用子类中定义但依赖子类属性的方法,除非你明确知道这些属性在调用时已经初始化。如果必须如此,可以考虑将属性提升到父类,或者在子类构造函数中重写父类逻辑以确保属性的正确初始化。

5. 总结

立即调用类表达式是一种强大的JavaScript模式,它允许我们以简洁的方式定义并实例化类。然而,当与继承结合使用时,尤其是在父类构造函数中调用子类方法并依赖子类属性时,其执行时序可能导致意想不到的结果(如 undefined)。

理解JavaScript类的声明、继承链、构造函数调用顺序以及实例属性的初始化时机是避免此类问题的关键。通过将依赖的属性提升到父类,或者重新设计方法调用逻辑,可以确保代码的正确性和可预测性。在复杂的类结构中,清晰地规划属性和方法的声明位置,是编写健壮、可维护代码的重要一环。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

552

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

374

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

731

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

477

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

394

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

990

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

656

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

551

2023.09.20

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

36

2026.01.14

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 3.6万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 2.2万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 2.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号