聊一聊 CommonJS 和 ES6 Module

1、为什么使用模块化?

模块化是指将一个复杂的系统分解为多个模块以方便编码。其实对于我们现在实际写的前端项目来说,文件即模块,各种模块(文件)构成我们的项目,然后模块之前通过一定的规则互相引用,然后在打包阶段,打包工具比如webpackrollup等会在打包阶段去处理好我们模块的引用关系。

在以前古老的开发模块中,开发网页要通过命令空间 的方式来组织代码,比如著名的 JQuery 库将它所有的 API 都挂到了 window.$下,这样会存在一些问题:

  • 命令空间冲突,两个库可能会使用同一个名称,例如 Zepto
  • 无法合理地管理项目的依赖和版本。
  • 无法方便地控制依赖的加载顺序。

一旦项目过大,这种方式会变得难以维护,所以后面模块化的思想就诞生了。

2、commonjs

2.1 特点

  • 动态引入,执行时引入。
  • 在运行后才可以得知模块导出内容,编译阶段无法做静态分析。
  • 输出的是值的拷贝。

2.2 commonjs 原理

我们先新建一个a.js文件,内容如下:

js 复制代码
// a.js
module.exports = 'a'

在同一个目录下新建一个b.js文件:

js 复制代码
// b.js
const fs = require('fs')
const path = require('path')
function req(targetPath) {
  const absPath = path.resolve(__dirname, targetPath)
  const content = fs.readFileSync(absPath, 'utf8')
  const module = {
    exports: {}
  }
  const fn = new Function('exports', 'module', 'require', '__dirname', '__filename', content + '\r\n return module.exports;')
  return fn.call(module.exports, module.exports, req, __dirname, __filename)
}
console.log(req('./a.js')) // 打印a

核心原理就是利用node原生的fs模块去读取文件,拿到代码字符串,然后通过new Function包装成一个函数去处理,然后通过自己在函数尾部新加的一行return module.exports将执行后的结果抛出。

3、ES Module

  • 静态引入,编译时引入。
  • 只能作为模块顶层的语句出现,不能出现在 function 里面或者是 if 里面。
  • import 的模块名只能是字符串常量。
  • 不管 import 的语句出现的位置在哪里,在模块初始化的时候所有的 import 都必须已经导入完成。
  • 输出的是值的引用。

4、commonjs 和 es module 使用场景

  • 浏览器端代码 :使用 es2015 module(也就是ES6 Module),模块化使用灵活,且可充分利用tree shaking减少代码体积
  • 服务端 node :适合动态引入,一般不支持 tree-shakinges module,同时也并不需要考虑代码体积,所以使用 commonjs 模块规范,同时也可以拥有更好的 debug 支持,提高开发效率。

tree-shaking:

  • tree-shaking 可以利用 ES2015(es6) 模块语法静态分析的特性,删除没有使用的代码,对代码体积进行优化
  • webpack tree-shaking 启条件:
  • 使用 es2015 模块语法(import和export), require 不行。
  • 配合JS代码压缩插件插件,如 UglifyJSPlugin,TerserPlugin。
  • 去除 babel-loader模 块转换插件,不让 babel-loader 进行模块转换,以保留 export 和 import 关键字,然后让 webpack 来转换 。

小结

目前前端已经是模块化的天下了,学好ES modulecommonjs就已经基本够用了,上面介绍了它们各自的特点和使用场景,希望能帮助到大家去更好的运用它们!

相关推荐
Java后端的Ai之路2 小时前
【AI应用开发工程师】-Gemini写前端的一个坑
前端·人工智能·gemini·ai应用开发工程师
亿元程序员2 小时前
最近很火的一个拼图游戏,老板让我用Cocos3.8做一个...
前端
m0_748250032 小时前
C++ Web 编程
开发语言·前端·c++
切糕师学AI2 小时前
Vue 中的响应式布局
前端·javascript·vue.js
行者962 小时前
Flutter适配OpenHarmony:跨平台开发热门标签组件,从数据到交互的完整实现
前端·flutter·harmonyos·鸿蒙
晷龙烬2 小时前
Vue组件使用三步走:创建、注册、使用(Vue2/Vue3双版本详解)
前端·javascript·vue.js
前端 贾公子2 小时前
微信小程序webview访问的url从https变成http原因排查
前端
2501_948122632 小时前
React Native for OpenHarmony 实战:Steam 资讯 App 设置页面
javascript·react native·react.js·游戏·ecmascript·harmonyos
云和数据.ChenGuang2 小时前
fastapi无法在微软的edge上运行程序
前端·edge·fastapi