AMD CMD UMD 的一些基本知识

cjs的代表是commonjs,只能运行在node环境中,浏览器并不认识。所有需要webpack或者其他工程化工具

CMD代表库: Sea.js

AMD代表库: requirejs

commonjs打包话工具: browserify

各种规范支持: webpack

原生ES Moudle: vite

commonjs使用

javascript 复制代码
// main.js
const util = require('./util')

function doSomething() {
    console.log('Doing something...');
    util.doUtilStuff();
}
module.exports = {
  doSomething
}

// util.js
function doUtilStuff() {
  console.log('doUtilStuff')
}

exports.doUtilStuff = doUtilStuff

commonjs可以使用2种导出方式: module.exports 和 exports。

module.exports是整个文件模块的出口对象。exports是module.exports的引用,用来简化导出操作。

加载文件规则:

  1. 解析文件路径,如果是 本地模块则在本地查找引入,核心模块则在内置模块中加载

  2. .js : 解析为javascripte代码。 .json解析为json 对象。 .node: 编译后的二进制模块

  3. 加载的js文件,node会进行编译然后执行

  4. 加载的模块会进行缓存避免重复加载

循环引用:是指2或多个文件之间相互引用。这种情况会引发文件未初始化的问题。

javascript 复制代码
// a.js
const B = require('./B')
console.log('---- Start A ----')

const say = () => {
    console.log('say name ph', B.walk())
}

console.log('---- End A ----')

module.exports = {
    name: 'Module A',
    say
}

// b.js
const A = require('./A')
console.log('---- Start B ----')

const walk = () => {

    console.log('I can ', A.name, ' when I walk')
}

console.log('---- End B ----')

module.exports = {
    name: 'Module B',
    walk
}

(node:46206) Warning: Accessing non-existent property 'Symbol(Symbol.toStringTag)' of module exports inside circular dependency

(node:46206)警告:在循环依赖中访问模块导出的不存在的属性'Symbol(nodejs.util.inspect.custom)'(使用' node -trace-warnings...'以显示创建警告的位置)

引发问题:会使得引用一方的代码未初始化话就引入,从而报错。

解决方案:

  1. 重构代码
  2. **延迟加载
    **‌b.js 中的require('./A') 或者a.js中的require('./B') 放入调用时的函数中。让文件先完成初始化即可。
  3. 使用弱引用

commonjs的原理?为什么在文件中可以直接使用require 和 exports/module.exports?

在b.js 文件中加入 console.log('---- Start B ----', arguments.callee.toString()), 得到如下:

javascript 复制代码
function (exports, require, module, __filename, __dirname) {

  console.log('---- Start B ----', arguments.callee.toString())
  
  const walk = () => {
      const A = require('./A')
      console.log('I can ', A.name, ' when I walk')
  }
  
  console.log('---- End B ----')
  
  module.exports = {
      name: 'Module B',
      walk
  }
}

什么是UMD?

UMD(Universal Module Definition)是JavaScript模块的一种通用模块定义模式。它的产生背景主要是为了解决JavaScript模块化在不同环境中的兼容性问题。随着前端开发的复杂度增加,开发者需要编写能够在多种环境下运行的代码,包括浏览器端和服务器端。然而,不同的环境支持不同的模块化规范,如AMD(异步模块定义)、CommonJS等。为了编写能够在这些环境中无缝运行的代码,UMD应运而生。

javascript 复制代码
(function (root, factory) {
    if (typeof define === 'function' && define.amd) {
        // AMD环境
        define(['jquery'], factory);
    } else if (typeof exports === 'object') {
        // CommonJS环境
        module.exports = factory(require('jquery'));
    } else {
        // 全局变量环境
        root.myModule = factory(root.jQuery);
    }
}(this, function ($) {
    // 模块主体
    function privateMethod() {
        // 私有方法
    }

    function publicMethod() {
        // 公共方法
    }

    return {
        publicMethod: publicMethod
    };
}));
相关推荐
candyTong18 小时前
一觉醒来,大模型就帮我排查完页面性能问题
前端·javascript·架构
玩嵌入式的菜鸡19 小时前
网页访问单片机设备---基于mqtt
前端·javascript·css
前端一小卒19 小时前
我用 Claude Code 的 Superpowers 技能链写了个服务,部署前差点把服务器搞炸
前端·javascript·后端
豹哥学前端1 天前
用猜数字游戏,一口气掌握 JavaScript 核心知识点(附完整代码)
前端·javascript
忆往wu前1 天前
从0到1一步步拆解搭建,梳理一个 Vue3 简易图书后台全开发流程
前端·javascript·vue.js
shao9185161 天前
第3章(2)——使用Gradio JavaScript Client
javascript·node.js·cdn·gradio·job·events·playcode
光影少年1 天前
大屏页面,一次多个请求,请求加密导致 点击 全局时间选择器 时出现卡顿咋解决(面板收起会延迟1~2秒)
前端·javascript·vue.js·学习·前端框架·echarts·reactjs
Mr.mjw1 天前
vue中封装一个环形进度条组件,根据外部盒子大小自适应变化
前端·javascript·vue.js
无心使然1 天前
Openlayers调用ArcGis影像服务之一动态地图、地图切片(/exportImage)
前端·javascript·数据可视化
像我这样帅的人丶你还1 天前
前端监控体系与实践(二):全局监控
前端·javascript·vue.js