2024年前端最新面试题:Commonjs 和 Es Module区别(1),web前端面试题csdn

打开全栈工匠技能包-1小时轻松掌握SSR

两小时精通jq+bs插件开发

生产环境下如歌部署Node.js

开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】

网易内部VUE自定义插件库NPM集成

谁说前端不用懂安全,XSS跨站脚本的危害

webpack的loader到底是什么样的?两小时带你写一个自己loader

// id 为路径标识符

function require(id) {

/* 查找  Module 上有没有已经加载的 js  对象*/

const  cachedModule = Module._cache[id]

/* 如果已经加载了那么直接取走缓存的 exports 对象  */

if(cachedModule){

return cachedModule.exports

}

/* 创建当前模块的 module  */

const module = { exports: {} ,loaded: false , …}

/* 将 module 缓存到  Module 的缓存属性中,路径标识符作为 id */

Module._cache[id] = module

/* 加载文件 */

runInThisContext(wrapper(‘module.exports = “123”’))(module.exports, require, module, __filename, __dirname)

/* 加载完成 *//

module.loaded = true

/* 返回值 */

return module.exports

}

从上面我们总结出一次 require 大致流程是这样的;

  • require 会接收一个参数——文件标识符,然后分析定位文件,分析过程我们上述已经讲到了,加下来会从 Module 上查找有没有缓存,如果有缓存,那么直接返回缓存的内容。

  • 如果没有缓存,会创建一个 module 对象,缓存到 Module 上,然后执行文件,加载完文件,将 loaded 属性设置为 true ,然后返回 module.exports 对象。借此完成模块加载流程。

  • 模块导出就是 return 这个变量的其实跟 a = b 赋值一样, 基本类型导出的是值, 引用类型导出的是引用地址。

  • exports 和 module.exports 持有相同引用,因为最后导出的是 module.exports, 所以对 exports 进行赋值会导致 exports 操作的不再是 module.exports 的引用。

require 避免重复加载

从上面我们可以直接得出,require 如何避免重复加载的,首先加载之后的文件的 module 会被缓存到 Module 上,比如一个模块已经 require 引入了 a 模块,如果另外一个模块再次引用 a ,那么会直接读取缓存值 module ,所以无需再次执行模块。

对应 demo 片段中,首先 main.js 引用了 a.js ,a.js 中 require 了 b.js 此时 b.js 的 module 放入缓存 Module 中,接下来 main.js 再次引用  b.js ,那么直接走的缓存逻辑。所以 b.js 只会执行一次,也就是在 a.js 引入的时候。

require 避免循环引用

那么接下来这个循环引用问题,也就很容易解决了。为了让大家更清晰明白,那么我们接下来一起分析整个流程。

  • ① 首先执行 node main.js ,那么开始执行第一行 require(a.js)

  • ② 那么首先判断 a.js 有没有缓存,因为没有缓存,先加入缓存,然后执行文件 a.js (需要注意 是先加入缓存, 后执行模块内容);

  • ③ a.js 中执行第一行,引用 b.js。

  • ④ 那么判断 b.js 有没有缓存,因为没有缓存,所以加入缓存,然后执行 b.js 文件。

  • ⑤ b.js 执行第一行,再一次循环引用 require(a.js) 此时的 a.js 已经加入缓存,直接读取值。接下来打印 console.log('我是 b 文件'),导出方法。

  • ⑥ b.js 执行完毕,回到 a.js 文件,打印 console.log('我是 a 文件'),导出方法。

  • ⑦ 最后回到 main.js,打印 console.log('node 入口文件') 完成这个流程。

不过这里我们要注意问题:

  • 如上第 ⑤ 的时候,当执行 b.js 模块的时候,因为 a.js 还没有导出 say 方法,所以 b.js 同步上下文中,获取不到 say。

我用一幅流程图描述上述过程:

15.jpg

为了进一步验证上面所说的,我们改造一下 b.js 如下:

const say = require(‘./a’)

const  object = {

name:‘《React进阶实践指南》’,

author:‘我不是外星人’

}

console.log(‘我是 b 文件’)

console.log(‘打印 a 模块’ , say)

setTimeout(()=>{

console.log(‘异步打印 a 模块’ , say)

},0)

module.exports = function(){

return object

}

打印结果:

6.jpg

  • 第一次打印 say 为空对象。

  • 第二次打印 say 才看到 b.js 导出的方法。

那么如何获取到 say 呢,有两种办法:

  • 一是用动态加载 a.js 的方法,马上就会讲到。

  • 二个就是如上放在异步中加载。

我们注意到 a.js 是用 exports.say 方式导出的,如果 a.js 用 module.exports 结果会有所不同。至于有什么不同,为什么?我接下来会讲到。

4 require 动态加载

上述我们讲了 require 查找文件和加载流程。接下来介绍 commonjs 规范下的 require 的另外一个特性——动态加载

require 可以在任意的上下文,动态加载模块。我对上述 a.js 修改。

a.js

console.log(‘我是 a 文件’)

exports.say = function(){

const getMes = require(‘./b’)

const message = getMes()

console.log(message)

}

main.js

const a = require(‘./a’)

a.say()

  • 如上在 a.js 模块的 say 函数中,用 require 动态加载 b.js 模块。然后执行在 main.js 中执行 a.js 模块的 say 方法。

打印结果如下:

7.jpg

require 本质上就是一个函数,那么函数可以在任意上下文中执行,来自由地加载其他模块的属性方法。

5 exports 和 module.exports

系统分析完 require ,接下来我们分析一下,exports 和 module.exports,首先看一下两个的用法。

exports 使用

第一种方式:exportsa.js

exports.name = 《React进阶实践指南》

exports.author = 我不是外星人

exports.say = function (){

console.log(666)

}

引用

const a = require(‘./a’)

console.log(a)

打印结果:

8.jpg

  • exports 就是传入到当前模块内的一个对象,本质上就是 module.exports

问题:为什么 exports={} 直接赋值一个对象就不可以呢? 比如我们将如上 a.js 修改一下:

exports={

name:‘《React进阶实践指南》’,

author:‘我不是外星人’,

say(){

console.log(666)

}

}

打印结果:

9.jpg

理想情况下是通过 exports = {} 直接赋值,不需要在  exports.a = xxx  每一个属性,但是如上我们看到了这种方式是无效的。为什么会这样?实际这个是 js 本身的特性决定的。

通过上述讲解都知道 exports , module 和 require 作为形参的方式传入到 js 模块中。我们直接 exports = {}  修改 exports ,等于重新赋值了形参,那么会重新赋值一份,但是不会在引用原来的形参。举一个简单的例子

function wrap (myExports){

myExports={

name:‘我不是外星人’

}

}

let myExports = {

name:‘alien’

}

wrap(myExports)

console.log(myExports)

打印:

10.jpg

我们期望修改 myExports ,但是没有任何作用。

假设 wrap 就是 Commonjs 规范下的包装函数,我们的 js 代码就是包装函数内部的内容。当我们把  myExports 对象传进去,但是直接赋值 myExports = { name:'我不是外星人' } 没有任何作用,相等于内部重新声明一份 myExports 而和外界的 myExports 断绝了关系。所以解释了为什么不能 exports={...} 直接赋值。

那么解决上述也容易,只需要函数中像 exports.name 这么写就可以了。

function wrap (myExports){

myExports.name=‘我不是外星人’

}

打印:

11.jpg

module.exports 使用

module.exports 本质上就是 exports ,我们用 module.exports 来实现如上的导出。

module.exports ={

name:‘《React进阶实践指南》’,

author:‘我不是外星人’,

say(){

console.log(666)

}

}

module.exports 也可以单独导出一个函数或者一个类。比如如下:

module.exports = function (){

// …

}

从上述 require 原理实现中,我们知道了 exports 和 module.exports 持有相同引用,因为最后导出的是 module.exports 。那么这就说明在一个文件中,我们最好选择 exports 和 module.exports 两者之一,如果两者同时存在,很可能会造成覆盖的情况发生。比如如下情况:

exports.name = ‘alien’ // 此时 exports.name 是无效的

module.exports ={

name:‘《React进阶实践指南》’,

author:‘我不是外星人’,

say(){

console.log(666)

}

}

  • 上述情况下 exports.name 无效,会被 module.exports 覆盖。
Q & A

1 那么问题来了?既然有了 exports,为何又出了 module.exports?

答:如果我们不想在 commonjs 中导出对象,而是只导出一个类或者一个函数再或者其他属性的情况,那么 module.exports 就更方便了,如上我们知道 exports 会被初始化成一个对象,也就是我们只能在对象上绑定属性,但是我们可以通过 module.exports 自定义导出出对象外的其他类型元素。

let a = 1

module.exports = a // 导出函数

module.exports = [1,2,3] // 导出数组

module.exports = function(){} //导出方法

2 与 exports 相比,module.exports 有什么缺陷 ?

答:module.exports 当导出一些函数等非对象属性的时候,也有一些风险,就比如循环引用的情况下。对象会保留相同的内存地址,就算一些属性是后绑定的,也能间接通过异步形式访问到。但是如果 module.exports 为一个非对象其他属性类型,在循环引用的时候,就容易造成属性丢失的情况发生了。

四 Es Module


Nodejs 借鉴了 Commonjs 实现了模块化 ,从 ES6 开始, JavaScript 才真正意义上有自己的模块化规范,

Es Module 的产生有很多优势,比如:

  • 借助 Es Module 的静态导入导出的优势,实现了 tree shaking

  • Es Module 还可以 import() 懒加载方式实现代码分割。

在 Es Module 中用 export 用来导出模块,import 用来导入模块。但是 export 配合 import 会有很多种组合情况,接下来我们逐一分析一下。

导出 export 和导入 import

所有通过 export 导出的属性,在 import 中可以通过结构的方式,解构出来。

export 正常导出,import 导入

导出模块:a.js

const name = ‘《React进阶实践指南》’

const author = ‘我不是外星人’

export { name, author }

export const say = function (){

console.log(‘hello , world’)

}

导入模块:main.js

// name , author , say 对应 a.js 中的  name , author , say

import { name , author , say } from ‘./a.js’

  • export { }, 与变量名绑定,命名导出。

  • import { } from ‘module’, 导入 module 的命名导出 ,module 为如上的 ./a.js

  • 这种情况下 import { } 内部的变量名称,要与 export { } 完全匹配。

默认导出 export default

导出模块:a.js

const name = ‘《React进阶实践指南》’

const author = ‘我不是外星人’

const say = function (){

console.log(‘hello , world’)

}

export default {

name,

author,

say

}

导入模块:main.js

import mes from ‘./a.js’

console.log(mes) //{ name: ‘《React进阶实践指南》’,author:‘我不是外星人’, say:Function }

  • export default anything 导入 module 的默认导出。anything 可以是函数,属性方法,或者对象。

  • 对于引入默认导出的模块,import anyName from 'module', anyName 可以是自定义名称。

混合导入|导出

ES6 module 可以使用 export default 和 export 导入多个属性。

导出模块:a.js

export const name = ‘《React进阶实践指南》’

export const author = ‘我不是外星人’

export default  function say (){

console.log(‘hello , world’)

}

导入模块:main.js 中有几种导入方式:

第一种:

import theSay , { name, author as  bookAuthor } from ‘./a.js’

console.log(

theSay,     // ƒ say() {console.log(‘hello , world’) }

name,       // “《React进阶实践指南》”

bookAuthor  // “我不是外星人”

)

第二种:

import theSay, * as mes from ‘./a’

console.log(

theSay, // ƒ say() { console.log(‘hello , world’) }

mes // { name:‘《React进阶实践指南》’ , author: “我不是外星人” ,default:  ƒ say() { console.log(‘hello , world’) } }

)

  • 导出的属性被合并到 mes 属性上, export 被导入到对应的属性上,export default 导出内容被绑定到 default 属性上。theSay 也可以作为被 export default 导出属性。

重属名导入

import { bookName as name, say, bookAuthor as author } from ‘module’

console.log( bookName , bookAuthor , say ) //《React进阶实践指南》 我不是外星人

  • 从 module 模块中引入 name ,并重命名为 bookName ,从 module 模块中引入 author ,并重命名为 bookAuthor。然后在当前模块下,使用被重命名的名字。

重定向导出

可以把当前模块作为一个中转站,一方面引入 module 内的属性,然后把属性再给导出去。

export * from ‘module’ // 第一种方式

export { name, author, …, say } from ‘module’ // 第二种方式

export { bookName as name, bookAuthor as author, …, say } from ‘module’ //第三种方式

  • 第一种方式:重定向导出 module 中的所有导出属性, 但是不包括 module 内的 default 属性。

  • 第二种方式:从 module 中导入 name ,author ,say 再以相同的属性名,导出。

  • 第三种方式:从 module 中导入 name ,重属名为 bookName 导出,从 module 中导入 author ,重属名为 bookAuthor 导出,正常导出 say 。

无需导入模块,只运行模块

import ‘module’

  • 执行 module 不导出值  多次调用 module 只运行一次。

动态导入

const promise = import(‘module’)

  • import('module'),动态导入返回一个 Promise。为了支持这种方式,需要在 webpack 中做相应的配置处理。

ES6 module 特性

接下来我们重点分析一下 ES6 module 一些重要特性。

1 静态语法

ES6 module 的引入和导出是静态的,import 会自动提升到代码的顶层 ,import , export 不能放在块级作用域或条件语句中。

???错误写法一:

function say(){

import name from ‘./a.js’

export const author = ‘我不是外星人’

}

???错误写法二:

isexport &&  export const  name = ‘《React进阶实践指南》’

这种静态语法,在编译过程中确定了导入和导出的关系,所以更方便去查找依赖,更方便去 tree shaking (摇树) , 可以使用 lint 工具对模块依赖进行检查,可以对导入导出加上类型信息进行静态的类型检查。

import 的导入名不能为字符串或在判断语句,下面代码是错误的

???错误写法三:

import ‘defaultExport’ from ‘module’

let name = ‘Export’

import ‘default’ + name from ‘module’

2 执行特性

ES6 module 和 Common.js 一样,对于相同的 js 文件,会保存静态属性。

但是与 Common.js 不同的是 ,CommonJS 模块同步加载并执行模块文件,ES6 模块提前加载并执行模块文件,ES6 模块在预处理阶段分析模块依赖,在执行阶段执行模块,两个阶段都采用深度优先遍历,执行顺序是子 -> 父。

为了验证这一点,看一下如下 demo。

main.js

console.log(‘main.js开始执行’)

import say from ‘./a’

import say1 from ‘./b’

console.log(‘main.js执行完毕’)

a.js

import b from ‘./b’

console.log(‘a模块加载’)

export default  function say (){

console.log(‘hello , world’)

}

b.js

console.log(‘b模块加载’)

export default function sayhello(){

console.log(‘hello,world’)

}

  • main.js 和 a.js 都引用了 b.js 模块,但是 b 模块也只加载了一次。

  • 执行顺序是子 -> 父

效果如下:

12.jpg

3 导出绑定

不能修改import导入的属性

a.js

export let num = 1

export const addNumber = ()=>{

num++

}

main.js

import {  num , addNumber } from ‘./a’

num = 2

如果直接修改,那么会报错。如下所示:

属性绑定

所以可以在 main.js 中这么修改。

import {  num , addNumber } from ‘./a’

console.log(num) // num = 1

addNumber()

console.log(num) // num = 2

  • 如上属性 num 的导入是绑定的。

接下来对 import 属性作出总结:

  • 使用 import 被导入的模块运行在严格模式下。

  • 使用 import 被导入的变量是只读的,可以理解默认为 const 装饰,无法被赋值

  • 使用 import 被导入的变量是与原变量绑定/引用的,可以理解为 import 导入的变量无论是否为基本类型都是引用传递。

import() 动态引入

import() 返回一个 Promise 对象, 返回的 Promise 的 then 成功回调中,可以获取模块的加载成功信息。我们来简单看一下 import() 是如何使用的。

main.js

setTimeout(() => {

const result  = import(‘./b’)

result.then(res=>{

console.log(res)

})

}, 0);

b.js

export const name =‘alien’

export default function sayhello(){

console.log(‘hello,world’)

}

打印如下:

13.jpg

从打印结果可以看出 import()的基本特性。

  • import() 可以动态使用,加载模块。

  • import() 返回一个 Promise ,成功回调 then 中可以获取模块对应的信息。name 对应 name 属性, default 代表 export default 。__esModule 为 es module 的标识。

import() 可以做一些什么

动态加载

  • 首先 import() 动态加载一些内容,可以放在条件语句或者函数执行上下文中。

if(isRequire){

const result  = import(‘./b’)

}

懒加载

  • import() 可以实现懒加载,举个例子 vue 中的路由懒加载;

[

{

path: ‘home’,

name: ‘首页’,

component: ()=> import(‘./home’) ,

},

]

React中动态加载

总结

三套“算法宝典”

开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】

28天读完349页,这份阿里面试通关手册,助我闯进字节跳动

算法刷题LeetCode中文版(为例)

人与人存在很大的不同,我们都拥有各自的目标,在一线城市漂泊的我偶尔也会羡慕在老家踏踏实实开开心心养老的人,但是我深刻知道自己想要的是一年比一年有进步。

最后,我想说的是,无论你现在什么年龄,位于什么城市,拥有什么背景或学历,跟你比较的人永远都是你自己,所以明年的你看看与今年的你是否有差距,不想做咸鱼的人,只能用尽全力去跳跃。祝愿,明年的你会更好!

由于篇幅有限,下篇的面试技术攻克篇只能够展示出部分的面试题,详细完整版以及答案解析,有需要的可以关注

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值