Javascript模块化开发基础,bilibili面试题

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Web前端全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上前端开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip1024c (备注前端)
img

正文

// 使用简写, 可读性不好,不建议

export {a, b, c} from ‘./abc’;

//ES7 提议,在简化先输入后输出的写法。现在不能使用,也不建议使用,可读性不好

export a, b, c from ‘./abc’

使用 import 和 export 需要注意一下几个方面:

  • export 必须写在所在模块作用于的顶层。如果写在了内部作用于会报错

  • export 输出的值是动态绑定的,绑定在其所在的模块。

// foo.js

export var foo = ‘foo’;

setTimeout(function() {

foo = ‘foo2’;

}, 500);

// main.js

import * as m from ‘./foo’;

console.log(m.foo); // foo

setTimeout(() => console.log(m.foo), 500); //foo2 500ms 后同样会被修改

  • import 具有声明提升,而且会提升到整个文件最上面

  • import 获得的变量都是只读的,修改它们会报错

  • 在 export 输出内容时,如果同时输出多个变量,需要使用大括号{},同时 import 导入多个变量也需要大括号

  • import 引入模块的默认后缀是 .js, 所以写的时候可以忽略 js 文件扩展名

  • import 会执行要所加载的模块。如下写法仅仅执行一个模块,不引入任何值

import ‘./foo’; //执行 foo.js 但不引入任何值

模块整体加载

当然模块可以作为整体加载,使用*关键字,并利用 as 重命名得到一个对象,所有获得的 export 的函数、值和类都是该对象的方法:

// abc.js

export var a = 1;

export var b = 2;

export var c = 3;

// main.js

import * as abc from ‘./abc’;

console.log(abc.a, abc.b, abc.c);

上面 main.js 中的整体加载可以用 module 关键字实现:

//暂时无法实现

module abc from ‘./abc’;

console.log(abc.a, abc.b, abc.c); //1 2 3

注意,以上2种方式获得的接口,不包括 export default 定义的默认接口。

export default

为了使模块的用户可以不看文档,或者少看文档,输出模块的时候利用 export default 指定默认输出的接口。使用 export defalut 输出时,不需要大括号,而 import 输入变量时,也不需要大括号(没有大括号即表示获得默认输出)

// abc.js

var a = 1, b = 2, c = 3;

export {a, b};

export default c; //等价于 export default 3;

// main.js

import {a, b} from ‘./abc’;

import num from ‘./abc’; // 不需要大括号, 而且可以直接改名(如果必须用原名不还得看手册么?)

console.log(a, b, num) // 1 2 3

本质上,export default输出的是一个叫做default的变量或方法,输入这个default变量时不需要大括号。

// abc.js

var a = 20;

export {a as default};

// main.js

import a from ‘./abc’; // 这样也是可以的

console.log(a); // 20

// 这样也是可以的

import {default as aa} from ‘./abc’;

console.log(aa); // 20

如果需要同时输入默认方法和其他变量可以这样写 import:

import customNameAsDefaultExport, {otherMethod}, from ‘./export-default’;

这里需要注意:一个模块只能有一个默认输出,所以 export default 只能用一次

模块的继承

所谓模块的继承,就是一个模块 B 输出了模块 A 全部的接口,就仿佛是 B 继承了 A。利用 export * 实现:

// circleplus.js

export * from ‘circle’; //当然,这里也可以选择只继承其部分接口,甚至可以对接口改名

export var e = 2.71828182846;

export default function(x){ //重新定义了默认输出,如果不想重新定义可以:export customNameAsDefaultExport from ‘circle’;

return Math.exp(x);

}

//main.js

import * from ‘circleplus’; //加载全部接口

import exp from ‘circleplus’; //加载默认接口

//…use module here

上面这个例子 circleplus 继承了 circle。值得一提的是,export * 不会再次输出 circle 中的默认输出(export default)。

在使用和定义模块时,希望可以做到以下几个建议:

  • Module 语法是 JavaScript 模块的标准写法,坚持使用这种写法。使用 import 取代 require, 使用 export 取代module.exports

  • 如果模块只有一个输出值,就使用 export default,如果模块有多个输出值,就不使用 export default

  • 尽量不要 export default 与普通的 export 同时使用

  • 不要在模块输入中使用通配符。因为这样可以确保你的模块之中,有一个默认输出(export default)

  • 如果模块默认输出一个函数,函数名的首字母应该小写;如果模块默认输出一个对象,对象名的首字母应该大写

ES6 模块加载的实质

ES6 模块加载的机制是值的应用,而 CommonJS 是值的拷贝。这意味着, ES6 模块内的值的变换会影响模块外对应的值,而 CommonJS 不会。 ES6 遇到 import 时不会立刻执行这个模块,只生成一个动态引用,需要用的时候再去里面找值。有点像 Unix 中的符号链接。所以说 ES6的模块是动态引用,不会缓存值。之前的这个例子就可以说明问题:

// foo.js

export let counter = 3;

export function inc(){

counter++;

}

// main.js

import {counter, inc} from ‘./foo’;

console.log(counter); //3

inc();

console.log(counter); //4

我们看一个 CommonJS 的情况

// foo.js

let counter = 3;

function inc(){

counter++;

}

module.exports = {

counter: counter,

inc: inc

}

// main.js

let foo = require(‘./foo’)

let counter = foo.counter;

let inc = foo.inc;

console.log(counter); //3

inc();

console.log(counter); //3

循环加载

不知道你们只不知道循环引用,在 js 垃圾回收机制中提到过:如果 A 对象的一个属性值是 B 对象,而 B 对象的一个属性值是 A 对象,就会形成循环引用,无法释放他们的内存。而模块中也会出现循环加载的情况:如果 A 模块的执行依赖 B 模块,而 B 模块的执行依赖 A 模块,就形成了一个循环加载,结果程序不能工作,或者死机。然而,这样的关系很难避免,因为开发者众多,谁都会在开发自己的模块时使用别人的几个模块,久而久之,就行互联网一样,这样的依赖也织成了一个网。

ES6 和 CommonJS 处理循环加载又不一样,从 CommonJS 开始研究

  • CommonJS

CommonJS 每次执行完一个模块对应的 js 文件后在内存中就生成一个对象:

{

id: ‘…’, //表示属性的模块名

exports: {…}; //模块输出的各个接口

loaded: true, //表示是否加载完毕

//…内容很多,不一一列举了

}

之后使用这个模块,即使在写一遍 requrie,都不会再执行对应 js 文件了,会直接在这个对象中取值。

CommonJS 如果遇到循环加载,就输出已执行的部分,之后的不再执行,执行顺序以注释序号为准(从0开始):

// a.js

exports.done = false; //1. 先输出 done

var b = require(‘./b.js’); //2. 进入 b.js 执行 b.js //5. 发现 a.js 没执行完,那就重复不执行 a.js,返回已经执行的 exports

console.log(In a.js, b.done = ${b.done}); //10. 第2步的 b.js 执行完了,继续执行 a.js 得到控制台输出:‘In a.js, b.done = true’

exports.done = true; //11

console.log(‘a.js executed’); //12. 得到控制台输出:“a.js executed”

// b.js

exports.done = false; //3. 先输出 done

var a = require(‘./a.js’); //4. 执行到这里发生循环加载,去 a.js 执行 a.js //6. 只得到了 a.js 中的 done 为 false

console.log(In b.js, a.done = ${a.done}); //7. 得到控制台输出:“In b.js, a.done = false”

exports.done = true; //8. 输出 done, 覆盖了第3步的输出

console.log(‘b.js executed’); //9. 得到控制台输出:“b.js executed”

//main.js

var a = require(“./a.js”); //0. 去 a.js 执行 a.js

var b = require(“./b.js”); //13. b.js 已经执行过了,直接去内存中的对象取值

console.log(In main,a.done = ${a.done}, b.done = ${b.done}) //得到控制台输出:‘In main,a.done = true, b.done = true’

  • ES6

由于 ES6 使用的是动态引用,遇到 import 时不会执行模块。所以和 CommonJS 有本质的区别。同样我们看个例子:

// a.js

更多面试题

**《350页前端校招面试题精编解析大全》**内容大纲主要包括 HTML,CSS,前端基础,前端核心,前端进阶,移动端开发,计算机基础,算法与数据结构,项目,职业发展等等

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024c (备注前端)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

会执行模块。所以和 CommonJS 有本质的区别。同样我们看个例子:

// a.js

更多面试题

**《350页前端校招面试题精编解析大全》**内容大纲主要包括 HTML,CSS,前端基础,前端核心,前端进阶,移动端开发,计算机基础,算法与数据结构,项目,职业发展等等

[外链图片转存中…(img-oxn9Falt-1713685452331)]

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024c (备注前端)
[外链图片转存中…(img-JDWCXX70-1713685452331)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值