当前位置: 首页 > news >正文

js模块化 --- commonjs规范 原理详解

什么是commonjs规范

        commonjs是一种模块化规范(nodejs的默认模块化规范,新版的nodejs已经支持es6的模块化,但它默认任然使用的是commonjs),通俗的说它将代码分割成了一个一个的模块,让不同的模块拥有自己独立的作用域(类似命名空间的隔断),在不同的模块中实现了重复命名的效果,同时支持模块的导入导出,让模块与模块之间可以相互调用,实现代码的复用。

tips:模块化可以让代码分离开和减少命名冲突问题,便于维护和开发,

模块化的核心:导入导出,作用域隔离

commonjs的导入导出

作为模块,最明显的特征就是支持导入导出,在commonjs中,使用require,module.exports,exports,来导入和导出。

const obj = {name: 'a',desc:'这是a模块'
}module.exports = obj
const obj = {name: 'b',desc: '这是b模块'
}module.exports = obj;
// 导入模块
const a = require('./a')
const b = require('./b')console.log(a)
console.log(b)

以上是a,b,index模块,一个js文件就是一个模块,在a,b模块中各自有一个obj对象导出,这里就实现了模块的命名分割,在不同的模块使用声明同名的变量名。然后在index中导入并使用

可以看到成功的拿到了各自模块中的obj对象。

这就是commonjs的基本使用,接下来分析一下它的模块化原理

commonjs的模块化原理

为什么可以在模块(js文件)中直接使用require,和module.exports,exports进行导入导出?

从用法上看很明显,require是一个函数,它接受一个模块(js文件)路径,返回这个模块的导出的值,而module.export,export是一个对象或者变量,它用来接受值作为模块的导出;也就是说,commonjs使用一个函数进行导入,一个对象或者变量进行导出;

tips:这里分析一下导入和导出的细节,

导入:获取一个模块导出的值

导出:将模块中的值暴露出去(丢给导入) 

        

分析一下, 这个过程和函数的返回调用非常相似,在函数中,

1.你可以返回一个值作为函数的返回值(导出),

2.同时可以调用函数获取到它的返回值(导入),

3.同时在不去调用函数的情况下,外部是没有办法拿到函数内部的东西(隔离),

4.函数可以重复调用(模块也可以重复导入)

在commonjs中,一个模块就是一个js文件,同时,一个模块也是一个函数,到这里可能会有点疑惑,这里我们直接在任意一个模块中打印argument

console.log( arguments)

在js函数中,出来箭头函数外都有一个argument参数,它包含了这个函数调用时传入的参数

可以看到一共有5个参数,这样可以还是不够清晰,在打印一下argument的callee属性

const obj = {name: 'a',desc:'这是a模块'
}module.exports = obj// arguments.callee 函数本身
console.log( arguments.callee.toString())

这下可以明显的看到在a模块的代码全都被放到了一个函数中,这个函数有5个参数exports, require, module, __filename, __dirname,这就是可以直接才模块中直接使用exports, require, module导入导出的原因。

所以在commonjs规范中,所以的js文件都作为模块,而对每一个js文件套上一层函数,是实现模块的原理,在不同的模块(函数)中,由于没有互相调用这个顶层函数,所以互相不能够访问到内部的数据。

到这里可以总结一下commonjs的原理了:

commonjs是通过给每个js文件‘外套’一层函数实现的模块化,即通过参数来导入导出,函数间作用域的隔断,函数的重复调用;但是这种模块化是基于运行时的,也就是说,虽然编译时外套了一层函数,但代码运行的时候才会进行导入导出(这很好理解,每个模块都是函数,调用模块也是在模块内调用的,函数运行了在能执行到导读导出的位置,将导出的值封装再对象中);

 

以上就是commonjs规范的模块化的原理,下面介绍一下有关commonjs的细节问题

和es6的module模块化规范的区别

这主要从以下几个方面区分:

1.ESmodule,是通过import,export,export default 进行导入导出的

2.ESmodule,是官方的规范,浏览器和node环境下都实现了对它的支持,

3.ESmodule,的模块化是基于编译时,在代码还没有执行时就已经进行了模块的导入导出,输出对应的接口,这是基于底层实现的

上面的第2点中,commonjs只支持在node环境下直接使用,浏览器种需要通过Babel进行转译才能使用,而ESmodule要在浏览器‘type = module’ 的script标签,node环境下,在package.json文件中配置,'type':'module'字段就可以使用ESmodule的规范(默认是'type':'commonjs')

 

ESMCJS
导入导出import,export default,exportrequire,module.export,exports
实现基于编译时基于运行时

 

 

 

 

 

 

 有关ESmodule的使用可以参考:js模块(module)和导入,导出(import,export)_js import module-CSDN博客

导出的优先级

使用commonjs规范的导入导出时,明显注意到,导入只有require,而导出可以使用exports,和module.exports;如果同时存在这两个导出,最终会是什么结果呢?

const obj = {name: 'a',desc:'这是a模块'
}// module.exports = obj
exports.desc = obj.desc;
exports.name = obj.name;// arguments.callee 函数本身
// console.log( arguments.callee.toString())
const obj = {name: 'b',desc: '这是b模块'
}module.exports = obj;const a = require('./a')console.log(a); // { name: 'a', desc: '这是a模块' }

a模块使用exports导出,然后在b模块导入a并运行b.js可以看到结果和module.exports的导出方法是一样的,都是将值赋给对象导出;

如果同时存在module.export和exports导出,最终会以module.exports为准,因为在commonjs实现的require函数中,最终返回的是module.exports ,在一个模块中this和export, module.exports的优先级关系(module.exports--->exports,this)

console.log(this,exports,module.exports)this.val = 'this';
exports.val = 'exports';
module.exports = {val:'module.exports'} ;console.log(this,exports,module.exports)

可以看到最开始module.exports,exports, this都是一个空对象(可以去改属性,但是不能去修改这个对象,否则导入会产生问题),因为this和exports是指向的同一个对象,所以,exports的赋值覆盖了this

console.log(this,exports,module.exports)exports.val = 'exports';
this.val = 'this';
module.exports = {val:'module.exports'} ;console.log(this,exports,module.exports)

颠倒一下顺序很容易就能看出来,而module.exports不一样,如果有单独导出值,那么它会以导出的值为准,若没有,则会以this和exports的值为准

大致就是:如果module.exports有值,则返回module.exports,否则,会将this,export的值赋值给module.exports,再返回module.exports;

下面测试一下导入的结果

index.js

exports.val = 'exports';
this.val = 'this';
module.exports = {val:'module.exports'} ;

b.js 


const index = require('./index')console.log(index);

 

 index.js


exports.val = 'exports';
this.val = 'this';
// module.exports = {val:'module.exports'} ;

b.js  


const index = require('./index')console.log(index);

 

可以看到,最终就是区分有没有给module.exports赋值,有的话以module.exports为准,没有的话,吧this,exports(this和exports是同一个对象,this === export)的值赋给module.exports,再以module.exports为准

相关文章:

js模块化 --- commonjs规范 原理详解

什么是commonjs规范 commonjs是一种模块化规范(nodejs的默认模块化规范,新版的nodejs已经支持es6的模块化,但它默认任然使用的是commonjs),通俗的说它将代码分割成了一个一个的模块,让不同的模块拥有自己独…...

kubeadm部署 Kubernetes(k8s) 高可用集群【V1.28 】

kubeadm是官方社区推出的一个用于快速部署kubernetes集群的工具。 calico.yaml kubernertes-dashboard.yaml 1. 安装要求 在开始之前,部署Kubernetes集群机器需要满足以下几个条件: 10台机器,操作系统Openeuler22.03 LTS SP4硬件配置&…...

【MySQL】MySQL Workbench下载安装、环境变量配置、基本MySQL语句、新建Connection

1.MySQL Workbench 下载安装: 进入网址:MySQL :: MySQL Workbench Manual :: 2 Installation (1)点击“MySQL Workbench on Windows”(下载Windows版本)(2)点击“Installing” &…...

CrowdStrike 的失败如何凸显了左移测试的重要性

通过自动化软件测试并将其左移,组织可以显著降低 CrowdStrike 等事件发生的风险。继续阅读,了解采用左移测试方法的强大之处。 Parasoft下载 测试中偷工减料的风险 CrowdStrike 软件更新失败是一个重要的教训,它让我们认识到早期、自动…...

HarmonyOS开发实战( Beta5版)高负载组件的渲染实践规范

简介 在应用开发中,有的页面需要在列表中加载大量的数据,就会导致组件数量较多或者嵌套层级较深,从而引起组件负载加重,绘制耗时增长。虽然可以通过组件复用避免组件重复创建,但是如果每个列表项中包含的组件较多&…...

NLP从零开始------16.文本中阶处理之序列到序列模型(1)

1. 序列到序列模型简介 序列到序列( sequence to sequence, seq2seq) 是指输入和输出各为一个序列(如一句话) 的任务。本节将输入序列称作源序列,输出序列称作目标序列。序列到序列有非常多的重要应用, 其中最有名的是机器翻译( machine translation), 机…...

【匈牙利汽车产业考察,开启新机遇】

匈牙利汽车工业发展历史悠久,拥有发达的基础设施和成熟的产业基础,全球20大汽车制造厂商中,有超过14家在匈牙利建立整车制造工厂和汽车零部件生产基地,比亚迪、宁德时代、欣旺达、蔚来等企业纷纷入驻。匈牙利位于东西方交汇处&…...

并行程序设计基础——动态进程管理

目录 一、组间通信域 二、动态创建新的MPI进程 1、MPI_COMM_SPAWN 2、MPI_COMM_GET_PARENT 3、MPI_COMM_SPAWN_MULTIPLE 三、独立进程间的通信 1、MPI_OPEN_PORT 2、MPI_COMM_ACCEPT 3、MPI_CLOSE_PORT 4、MPI_COMM_CONNECT 5、MPI_COMM_DISCONNECT 6、MPI_PUBLISH…...

C# 字符串(String)使用教程

在 C# 中,您可以使用字符数组来表示字符串,但是,更常见的做法是使用 string 关键字来声明一个字符串变量。string 关键字是 System.String 类的别名。 创建 String 对象 您可以使用以下方法之一来创建 string 对象: 通过给 Str…...

django之ForeignKey、OneToOneField 和 ManyToManyField

在Django中,ForeignKey、OneToOneField 和 ManyToManyField 是用于定义模型之间关系的字段类型。 ForeignKey ForeignKey 用于定义多对一的关系。例如,一个Employee可以属于一个Department,一个Department可以有多个Employee。 from djang…...

java.lang.IndexOutOfBoundsException: setSpan ( 0...x ) ends beyond length X

1,可能是EditText,setSelection(x)时超过了 输入框内容的实际长度导致的。 2,手机开启“拼写检查功能”,EditText设置了最大长度,选择提示的某一项文案时超过设置的最大长度限制,导致崩溃。 针对情况2 开…...

技术进展:CH-90树脂在去除硫酸钠柠檬酸钠溶液中铁锰离子上的应用

随着环境保护法规的日趋严格,以及工业生产中对产品纯度要求的不断提高,去除废水中的重金属离子已成为一个亟待解决的问题。铁和锰作为常见的杂质离子,在电池制造等行业中,对溶液纯度的影响不容忽视。 三元前驱体废水中通常含有硫…...

录屏时摄像头无法识别?如何录屏时打开摄像头,解决方案及录屏软件推荐

在数字时代,无论是游戏玩家、在线教育者还是企业培训师,录屏软件都已成为日常工作和娱乐中不可或缺的工具。但有时候想录制人物摄像头画面的时候,当录屏软件无法识别到摄像头时,这无疑会给用户带来不小的困扰。本文将提供一系列解…...

达梦数据库-DM8 企业版安装指南

一、DM8 企业版简介 达梦数据库(DM8)是中国自主研发的一款高性能数据库管理系统,广泛应用于企业级应用场景。DM8 企业版具备高可用性、强一致性和高性能等特点,支持多种操作系统和硬件平台。本文将详细介绍如何在 Kylin 操作系统上安装达梦数据库 DM8 企业版。 二、安装前…...

心脑血管科董田林医生:心律失常患者饮食,调养秘诀,助你找回健康心跳

在纷繁复杂的健康议题中,心律失常作为一种常见的心脏疾病,不仅影响着患者的生活质量,更牵动着每一个家庭的神经。幸运的是,通过科学合理的饮食调养,心律失常患者可以在很大程度上改善病情,逐步找回健康的心…...

期权杂记(一)

2024年9月5日: 切忌裸奔!如果你想暴富,押注期权还不如去澳门;做任何策略都可以多多关注希腊字母;对冲也是又方向性的,可以偏购,也可以偏沽,通过Delta Money来尝试计算;单…...

【MATLAB源码-第163期】基于matlab的BPSK+瑞利(rayleigh)信道下有无波束成形误码率对比仿真。

操作环境: MATLAB 2022a 1、算法描述 在通信系统中,波束成形(Beamforming)技术是一种广泛使用的信号处理技术,通过调整天线阵列中各个元素的相位和幅度,使得信号在特定方向上增强,在其他方向…...

【数据分享】2000-2022年我国省市县三级的逐日O3数据(免费获取\excel\shp格式)

空气质量数据是在我们日常研究中经常使用的数据!之前我们给大家分享了2000-2022年的省市县三级的逐日PM2.5数据、2013-2022年的省市县三级的逐日CO数据、2013-2022年的省市县三级的逐日SO2数据、2008-2022年我国省市县三级的逐日NO2数据和2000-2022年我国省市县三级…...

Python 的http.server库详细介绍

http.server 是 Python 标准库中的一个模块,用于创建基本的 HTTP 服务器。这个模块非常适合用于开发、测试、以及在本地网络中共享文件。以下是对 http.server 模块的详细介绍。 Python 官方文档:http.server — HTTP 服务器 模块概述 http.server 提…...

使用ffmpeg在视频中绘制矩形区域

由于项目需要对视频中的人脸做定位跟踪, 我先使用了人脸识别算法,对视频中的每个帧识别人脸、通过人脸库比对,最终记录坐标等信息。 然后使用ffmpeg中的 drawbox 滤镜功能,选择性的绘制区域。从而实现人脸定位跟踪 1、drawbox …...

工业安全零事故的智能守护者:一体化AI智能安防平台

前言: 通过AI视觉技术,为船厂提供全面的安全监控解决方案,涵盖交通违规检测、起重机轨道安全、非法入侵检测、盗窃防范、安全规范执行监控等多个方面,能够实现对应负责人反馈机制,并最终实现数据的统计报表。提升船厂…...

Go 语言接口详解

Go 语言接口详解 核心概念 接口定义 在 Go 语言中,接口是一种抽象类型,它定义了一组方法的集合: // 定义接口 type Shape interface {Area() float64Perimeter() float64 } 接口实现 Go 接口的实现是隐式的: // 矩形结构体…...

Auto-Coder使用GPT-4o完成:在用TabPFN这个模型构建一个预测未来3天涨跌的分类任务

通过akshare库,获取股票数据,并生成TabPFN这个模型 可以识别、处理的格式,写一个完整的预处理示例,并构建一个预测未来 3 天股价涨跌的分类任务 用TabPFN这个模型构建一个预测未来 3 天股价涨跌的分类任务,进行预测并输…...

MySQL中【正则表达式】用法

MySQL 中正则表达式通过 REGEXP 或 RLIKE 操作符实现(两者等价),用于在 WHERE 子句中进行复杂的字符串模式匹配。以下是核心用法和示例: 一、基础语法 SELECT column_name FROM table_name WHERE column_name REGEXP pattern; …...

[Java恶补day16] 238.除自身以外数组的乘积

给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂度…...

基于TurtleBot3在Gazebo地图实现机器人远程控制

1. TurtleBot3环境配置 # 下载TurtleBot3核心包 mkdir -p ~/catkin_ws/src cd ~/catkin_ws/src git clone -b noetic-devel https://github.com/ROBOTIS-GIT/turtlebot3.git git clone -b noetic https://github.com/ROBOTIS-GIT/turtlebot3_msgs.git git clone -b noetic-dev…...

C#中的CLR属性、依赖属性与附加属性

CLR属性的主要特征 封装性: 隐藏字段的实现细节 提供对字段的受控访问 访问控制: 可单独设置get/set访问器的可见性 可创建只读或只写属性 计算属性: 可以在getter中执行计算逻辑 不需要直接对应一个字段 验证逻辑: 可以…...

LLMs 系列实操科普(1)

写在前面: 本期内容我们继续 Andrej Karpathy 的《How I use LLMs》讲座内容,原视频时长 ~130 分钟,以实操演示主流的一些 LLMs 的使用,由于涉及到实操,实际上并不适合以文字整理,但还是决定尽量整理一份笔…...

搭建DNS域名解析服务器(正向解析资源文件)

正向解析资源文件 1)准备工作 服务端及客户端都关闭安全软件 [rootlocalhost ~]# systemctl stop firewalld [rootlocalhost ~]# setenforce 0 2)服务端安装软件:bind 1.配置yum源 [rootlocalhost ~]# cat /etc/yum.repos.d/base.repo [Base…...

【 java 虚拟机知识 第一篇 】

目录 1.内存模型 1.1.JVM内存模型的介绍 1.2.堆和栈的区别 1.3.栈的存储细节 1.4.堆的部分 1.5.程序计数器的作用 1.6.方法区的内容 1.7.字符串池 1.8.引用类型 1.9.内存泄漏与内存溢出 1.10.会出现内存溢出的结构 1.内存模型 1.1.JVM内存模型的介绍 内存模型主要分…...