JavaScript 模块化规范

122次阅读
没有评论

共计 2962 个字符,预计需要花费 8 分钟才能阅读完成。

概述

什么是模块化

  • 将程序文件依据一定规则 拆分 成多个文件,这种编码方式就是模块化的编码方式
  • 拆分出来的 每个文件就是一个模块,模块中的数据都是私有的,模块之间互相隔离
  • 同时也能通过一些手段,把模块内的指定数据“交出去”,供其他模块使用

为什么需要模块化

随着应用的复杂度越来越高,其代码量和文件数量都会急剧增加,会逐渐引发以下问题:

  1. 全局污染问题
  2. 依赖混乱问题
  3. 数据安全问题

有哪些模块化规范?

随着时间的推移,针对 JavaScript 的不同运行环境,相继出现了多种模块化规范,按时间排序,分别为:

  1. CommonJS 服务端应用广泛
  2. AMD
  3. CMD
  4. ES6 模块化 浏览器端应用广泛

导入与导出

模块化的核心思想是:模块之间是隔离的,通过导入和导出进行数据和功能的共享。

  • 导出(暴露):模块公开其内部的一部分(如变量、函数等),使这些内容可以被其他模块使用
  • 导入(引入):模块引入和使用其他模块导出的内容,以重用代码和功能

CommonJS 规范

导出数据

在 CommonJS 标准中,导出数据有两种方式:

  • 第一种方式:module.exports = value
  • 第二种方式:exports.name = value
// a.js
const name = '张三'

function getTel() {return '123456789'}

module.exports = {name, getTel}

// b.js
const name = '李四'

function getTel() {return '666666666'}

// 通过给 exports 对象添加属性的方式,来导出数据
exports.name = name
exports.getTel = getTel

// index.js
// 引入 a 模块暴露的所有内容
const a = require('./a')

// 引入 b 模块暴露的所有内容
const b = require('./b')

console.log(a, a.getTel())
console.log(b, b.getTel())

注意点:

  • 每个模块内部的 this、exports、modules.exports 在初始时,都指向同一个空对象,该空对象就是当前模块导出的数据
  • 无论如何修改导出对象,最终导出的都是 module.exports 的值
  • exports 是对 module.exports 的初始引用,仅为了方便给导出象添加属性,所以不能使用 exports = value 的形式导出数据,但是可以使用 module.exports = value 导出数据

导入数据

在 CJS 模块化标准中,使用内置的 require 函数进行导入数据。

// index.js
// 1 直接引入模块
const a = require('./a')
console.log(a, a.getTel())

// 2 引入同时解构出要用的数据
const {name, getTel} = require('./a')
console.log(name, getTel())

// 3 引入同时解构 + 重命名
const {name: stuName, getTel: stuTel} = require('./a')
console.log(stuName, stuTel())

扩展理解

一个 JS 模块在执行时,是被包裹在一个内置函数中执行的,所以每个模块都有自己的作用域,可通过如下方式来验证这一说法:

// index.js
console.log(arguments.callee.toString())

内置函数大致形式如下:

function (exports, require, module, __filename, __dirname) {
// index.js
console.log(arguments.callee.toString())

}

浏览器端运行

Node.js 默认支持 CommonJS 规范,但浏览器端不支持,所以需要经过编译,步骤如下:

  • 第一步:全局安装 browserify:pnpm i browserify -g
  • 第二步:编译:browserify index.js -o build.js,index.js 是源文件,build.js 是输出目标文件
  • 第三步:在 index.html 页面中引入使用:<script type="text/javascript" src="./build.js"></script>

ES6 模块化规范

ES6 模块化规范是一个官方标准的规范,它是在语言标准的层面上实现了模块化功能,是目前最流行的模块化规范,且浏览器与服务端均支持该规范。

运行 ES6 模块

浏览器端可以直接运行,如在 index.html 页面中引入使用:<script type="module" src="./index.js"></script>

在 Node.js 中运行 ES6 模块代码有两种方式:

  • 方式一:将 js 文件后缀从.js 改为.mjs,Node 会自动识别 ES6 模块
  • 方式二:在 package.json 文件中设置 type 属性值为 module

导出数据

ES6 模块化提供 3 种导出方式:分别导出、统一导出、默认导出。这 3 种导出方式,可以同时使用。

使用原则:导出的常量,务必用 const 定义。

分别导出:

// a.js 分别导出
export const name = '张三'

export function getTel() {return '123456789'}

统一导出:

// b.js 统一导出
const name = '李四'

function getTel() {return '666666666'}

export {name, getTel}

默认导出:

// c.js 默认导出
const name = '王五'
export const age = 25

function getTel() {return '999999999'}

export default {name, getTel}

导入数据

对于 ES6 模块化来说,使用何种导入方式,要根据导出方式决定。

「导入全部」(通用),可将模块中的所有导出内容整合到一个对象中。

// index.js
import * as a from './a.js'

console.log(a, a.getTel())

「命名导入」(对应导出方式:分别导出、统一导出)

// index.js 命名导入
import {name, getTel} from './a.js'
import {name as stuName, getTel as stuTel} from './a.js'

console.log(name, getTel())
console.log(stuName, stuTel())

「默认导入」(对应导出方式:默认导出)

// index.js 默认导入
import c from './c.js' // 默认导出的名字可以修改,如把 c 改为 abc 也行

console.log(c.name, c.getTel())

「命名导入 与 默认导入 可以混用」,且默认导入的内容必须放在前方。

// index.js 默认导入
import c, {age} from './c.js'

console.log(c.name, c.getTel(), age)

「动态导入」(通用),允许在运行时按需加载模块,返回值是一个 Promise。

// index.js
const a = await import('./a.js')
console.log(a)

import 可以不接收任何数据,如只让 a.js 参与运行:import './a.js'

正文完
 0
三毛笔记
版权声明:本站原创文章,由 三毛笔记 于2024-07-31发表,共计2962字。
转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。
评论(没有评论)