Webpack入门,模块打包之CommonJS简介

CommonJS

说到前端的模块,不得不讲一下CommonJS。CommonJS是由JavaScript社区于2009年提出的包含模块、文件、IO、控制台在内的一系列标准。Node.js的实现中采用了CommonJS标准的一部分,并在其基础上进行了一些调整。真正的CommonJS模块和Node.js中实现并不完全一样,通常情况下,谈到的CommonJS其实是Node.js中的版本,而非它的原始定义。

CommonJS最初只为服务端而设计,直到有了Browserify——一个运行在Node.js环境下的模块打包工具,它可以将CommonJS模块打包为浏览器可以运行的单个文件。这意味着客户端的代码也可以遵循CommonJS标准来编写了。

何为模块

模块之于程序,就如同细胞之于生物体,是具有特定功能的组成单元。不同的模块负责不同的工作,它们以某种方式联系在一起,共同保证程序的正常运转。

CommonJS中规定每个文件是一个模块。将一个JavaScript文件直接通过script标签插入页面中于封装成CommonJS模块最大的区别在于,前者的顶层作用域是全局作用域,而进行变量及函数声明时会污染全局变量;而后者会形成一个属于模块自身的作用域,所有的变量及函数只有自己能访问,对外是不可见的。

举例为证

在工程中新建calculator.js文件与index.js文件

// calculator.js
var name = "calculator.js";
// index.js
var name = "index.js";
require("./calculator.js");
console.log(name);

这里有两个文件,在index.js中通过CommonJS的require函数加载calculator.js。运行之后控制台输出结果是“index.js”,说明calculator.js中的变量声明并不会影响index.js,可见每个模块是拥有各自的作用域的。

导出

导出是一个模块向外暴露自身的唯一方式。在CommonJS中,通过module.exports可以导出模块中的内容。

// calculator.js
module.exports = {
  name: "calculator",
  add: function(a,b) {
    return a + b;
  }
}

CommonJS模块内部会用一个module对象存放当前模块的信息,可以理解成在每个模块的最开始定义了以下对象:var module = {...}; module.exports = {...}

module.exports用来指定该模块要对外暴露的哪些内容,在上面的额代码中导出了一个对象,包含name和add两个属性。为了书写方便,CommonJS也支持另外一种简化的导出方式——直接使用exports

exports.name = "calculator.js";
exports.add = function(a,b) {
  return a + b;
}

在实际效果上,这段代码和上面的module.exports没有任何不同。其内在机制是将exports指向module.exports,而module.exports在初始化时是一个空对象。可以简单的理解为,CommonJS在每个模块的首部默认添加了以下代码

var module = {
  exports: {}
};
var exports = module.exports;

因此,为exports.add赋值相当于在module.exports对象上添加一个属性。

注意事项

在使用exports时,不要直接给exports赋值,否则会导致其失效。
不要混合使用module.exports与exports

导入

在commonJS中使用require语法进行模块导入。

// calculator.js
module.exports = {
  add: function(a,b) {
    return a + b;
  }
}
// index.js
const calculator = require("./calculator.js");
const sum = calculator.add(5,7);
console.log(sum);

在index.js中导入了calculator模块,并调用了它的add函数。

使用require导入一个模块时会有两种情况:

  • 该模块未曾被加载过。这时会首先执行该模块,然后获取到该模块最终导出的内容。
  • 该模块已经被加载过。这时该模块的代码不会再次执行,而是直接获取该模块上一次导出的内容。

有时在加载一个模块时,不需要获取其导出的内容,只是想要通过执行它而产生某种作用,比如把它的接口挂在全局对象上,此时直接使用require即可。

require函数可以接收表达式,借助这个特性我们可以动态地指定模块加载路径。

const moduleNames = ["foo.js","bar.js"];
moduleNames.forEach(name => {
  require('./' + name);
})

示例代码仓库

https://gitee.com/zero_79152105/webpack-vblog

原创文章,作者:ZERO,如若转载,请注明出处:https://www.edu24.cn/course/webpack-module-commonjs.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
ZEROZERO
上一篇 2022年11月10日
下一篇 2022年11月11日

相关推荐

  • 如何判断一个对象为数组

    使用 instanceof 操作符 原理 instanceof 的内部机制是通过判断对象的原型链中是不是能找到类型的 prototype。 使用 instanceof 判断一个对象…

    2020年7月3日
    1.1K
  • Webpack入门,自定义loader

    有时开发者会遇到现有loader无法很好满足需求的情况,这时就需要对其进行修改,或者编写新的loader。 下面以一个简单的示例,讲解以下如何实现一个loader。 需求 实现一个…

    2022年11月23日
    704
  • Java自学之类与对象

    定义 类是由成员属性和方法组成。成员属性主要定义类的一个具体信息,实际上一个成员属性就是一个变量,而方法是一些操作的行为。类的定义基础语法如下。 一个类定义完成后并不能够被直接使用…

    2020年11月25日
    1.2K
  • spring boot练习篇之用户登录系统【接入数据库】

    抛弃JSP,只做纯粹的前后端分离项目。 写在前面 学习基础知识是枯燥无味的,之所以会这样,多数是因为心不静,对于如何运用它,感到茫然。所以建议大家在学习Java基础知识的时候,一定…

    2021年5月28日
    1.2K
  • windows下使用Docker Desktop安装nacos与mysql,实现互通访问

    1、命令行拉取MySQL镜像 打开命令提示符,运行以下命令安装mysql 2、运行mysql镜像,启动mysql实例 3、命令行拉取nacos镜像 4、运行nacos镜像,启动na…

    2022年10月27日
    416
  • STS插件mybatis-generator安装及使用

    断断续续学习Java也有好长时间了,没有师傅带,没有项目练手,学习超级慢,也很烦。视频、书籍翻看了一大推,还是没有目标。 相信滴水成海,外加条条大路通罗马,只要坚持,自己终能达成目…

    2019年12月27日
    2.2K
  • Java自学之String类

    在实际项目开发中,String是一个必须使用的程序类,可以说是项目的核心组成类。在Java程序里所有的字符串都要求使用【’‘】进行定义,同时也可以利用【+】实现字符串的连接处理。 …

    2020年11月30日
    1.4K
  • css晦涩难懂的点都在这啦

    CSS大家肯定都是会的但是每个人所撑握的情况都不一样,特别是已经工作几年的前辈(这里指的是我司)很多CSS玩法都不知道,可能他们已经习惯了用组件, 但是面试的时候又不可避免问,所以…

    2021年1月20日
    1.3K
  • JAVA学习之多线程知识点整理

    1、什么是进程?什么是线程? 进程是一个应用程序。线程是一个进程中的执行场景或者执行单元。一个进程可以启动多个线程。进程之间内存独立不共享。同一个进程中的线程之间,堆内存和方法区内…

    2020年6月19日
    1.2K