【前端基础】ES6模块与单例模式

使用type=“module"属性的<script>html标签会告诉浏览器相关执行的代码是模块代码。
模块代码可以直接嵌入在网页中,也可以作为外部文件引入:
单例模式:单一的实例,确保只有一个实例,并提供全局访问。

<script type="module">
 // 模块代码
</script>
<!-- 或者 -->
<script type="module" src="path/to/myModule.js"></script>

注意:解析到<script type="module">标签后会立即下载模块文件,但执行会延迟到文档解析完成。
执行顺序:

<!-- 第二个执行 -->
<script type="module"></script>
<!-- 第三个执行 -->
<script type="module"></script>
<!-- 第一个执行 -->
<script></script>

一个页面上有多少个入口模块没有限制,重复加载同一个模块也没有限制。同一个模块无论在一个页面中被加载多少次,也不管它是如何加载的,实际上都只会加载(代码执行)一次。

<!-- moduleA 在这个页面上只会被加载一次 -->
<script type="module">
 import './moduleA.js'
<script>
<script type="module">
 import './moduleA.js'
<script>
<script type="module" src="./moduleA.js"></script>
<script type="module" src="./moduleA.js"></script> 

ES6模块的行为特性

  • 模块代码只能在加载后执行
  • 模块只能加载一次
  • 模块是单例的
  • 模块可以定义公共接口,其他模块可以基于这个公共接口观察和交互
  • 模块可以请求加载其他模块
  • 支持循环依赖
  • 默认在严格模式下执行
  • 不共享全局命名空间
  • 模块顶级的this不指向window,指向被导出的对象
  • 模块中var声明的变量不会添加到window对象中
  • 模块是异步加载和执行的

下面来看看实例代码吧

在vscode中创建以下目录文件:
在这里插入图片描述
module1.ts的代码如下:

//单例模式,单个实例, 模块只能加载一次
let instance: any = null;
console.log("🚀 ~ file: module1.ts ~ line 4 ~ this", this)
console.log("🚀 ~ file: module1.ts ~ line 4 ~ this",'test调用导入多次只执行一次');

class SingleI{
  protected name: string
  constructor(name: string){
    this.name = name;
  }

  public getName(){
    console.log(this.name);
  }

  static getInstance(name: string){
    if (!instance) {
      instance = new SingleI(name);
    }

    return instance;
  }
}

export {
  instance,
  SingleI
}

module2.ts的代码如下:

//单例模式与模块化

import { SingleI } from "./module1";
export const single2:SingleI = SingleI.getInstance('h2');
console.log("🚀 ~ file: module2.ts ~ line 5 ~ this", this)

module3.ts的代码如下:

//单例模式与模块化

import { SingleI } from "./module1";


export const single3 = SingleI.getInstance('h3');

main.ts的代码如下:

import { single3 } from './module3';
import { single2 } from './module2';
//单例模式与模块化

console.log(single2 === single3);
single2.getName()
single3.getName()

在终点命令行中输入:ts-node main.ts
得到如下结果:
在这里插入图片描述
模块1代码被模块2和模块3导入两次,但其代码只执行了1次。符合es6模块的行为特性,由于多次导入也只执行一次,所以

static getInstance(name: string){
    if (!instance) {
      instance = new SingleI(name);
    }

    return instance;
  }

这个类上的方法就只会生成一个实例,多次调用不会执行再执行instance = new SingleI(name);这条语句。

再看模块文件的this指向,可以发现模块1的this指向export { instance, SingleI } export导出的对象。其中的值与代码的顺序有关,而且可以发现export导出的对象可以变量提升。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

h沐

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值