Javascript模块化开发基础,最新美团点评前端团队面试题

简介: Javascript模块化开发基础,最新美团点评前端团队面试题
export {a, b, c}
//main.js
import {a, b, c} from ‘./abc’; //接受的变量用大括号表示,以解构赋值的形式获取
console.log(a, b, c);
导入的时候也可以为变量重新取一个名字
import {a as aa, b, c};
console.log(aa, b, c)
如果想在一个模块中先输入后输出同一个模块,import语句可以和export语句写在一起。
// 正常写法
import {a, b, c} form ‘./abc’;
export {a, b, c}
// 使用简写, 可读性不好,不建议
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
}
相关文章
|
13天前
|
JSON JavaScript 前端开发
[JS]面试官:你的简历上写着熟悉jsonp,那你说说它的底层逻辑是怎样的?
本文介绍了JSONP的工作原理及其在解决跨域请求中的应用。首先解释了同源策略的概念,然后通过多个示例详细阐述了JSONP如何通过动态解释服务端返回的JavaScript脚本来实现跨域数据交互。文章还探讨了使用jQuery的`$.ajax`方法封装JSONP请求的方式,并提供了具体的代码示例。最后,通过一个更复杂的示例展示了如何处理JSON格式的响应数据。
27 2
[JS]面试官:你的简历上写着熟悉jsonp,那你说说它的底层逻辑是怎样的?
|
5天前
|
资源调度 前端开发 JavaScript
vite3+vue3 实现前端部署加密混淆 javascript-obfuscator
【11月更文挑战第10天】本文介绍了在 Vite 3 + Vue 3 项目中使用 `javascript-obfuscator` 实现前端代码加密混淆的详细步骤,包括安装依赖、创建混淆脚本、修改 `package.json` 脚本命令、构建项目并执行混淆,以及在 HTML 文件中引用混淆后的文件。通过这些步骤,可以有效提高代码的安全性。
|
13天前
|
设计模式 前端开发 JavaScript
揭秘!前端大牛们如何巧妙利用JavaScript,打造智能交互体验!
【10月更文挑战第30天】前端开发领域充满了无限可能与创意,JavaScript作为核心语言,凭借强大的功能和灵活性,成为打造智能交互体验的重要工具。本文介绍前端大牛如何利用JavaScript实现平滑滚动、复杂动画、实时数据更新和智能表单验证等效果,展示了JavaScript的多样性和强大能力。
31 4
|
11天前
|
机器学习/深度学习 自然语言处理 前端开发
前端神经网络入门:Brain.js - 详细介绍和对比不同的实现 - CNN、RNN、DNN、FFNN -无需准备环境打开浏览器即可测试运行-支持WebGPU加速
本文介绍了如何使用 JavaScript 神经网络库 **Brain.js** 实现不同类型的神经网络,包括前馈神经网络(FFNN)、深度神经网络(DNN)和循环神经网络(RNN)。通过简单的示例和代码,帮助前端开发者快速入门并理解神经网络的基本概念。文章还对比了各类神经网络的特点和适用场景,并简要介绍了卷积神经网络(CNN)的替代方案。
|
11天前
|
移动开发 前端开发 JavaScript
前端实训,刚入门,我用原生技术(H5、C3、JS、JQ)手写【网易游戏】页面特效
于辰在大学期间带领团队参考网易游戏官网的部分游戏页面,开发了一系列前端实训作品。项目包括首页、2021校园招聘页面和明日之后游戏页面,涉及多种特效实现,如动态图片切换和人物聚合效果。作品源码已上传至CSDN,视频效果可在CSDN预览。
18 0
前端实训,刚入门,我用原生技术(H5、C3、JS、JQ)手写【网易游戏】页面特效
|
16天前
|
JavaScript 前端开发 开发者
前端框架对比:Vue.js与Angular的优劣分析与选择建议
【10月更文挑战第27天】在前端开发领域,Vue.js和Angular是两个备受瞩目的框架。本文对比了两者的优劣,Vue.js以轻量级和易上手著称,适合快速开发小型到中型项目;Angular则由Google支持,功能全面,适合大型企业级应用。选择时需考虑项目需求、团队熟悉度和长期维护等因素。
22 1
|
17天前
|
JavaScript 前端开发 API
前端框架对比:Vue.js与Angular的优劣分析与选择建议
【10月更文挑战第26天】前端技术的飞速发展让开发者在构建用户界面时有了更多选择。本文对比了Vue.js和Angular两大框架,介绍了它们的特点和优劣,并给出了在实际项目中如何选择的建议。Vue.js轻量级、易上手,适合小型项目;Angular结构化、功能强大,适合大型项目。
16 1
|
20天前
|
缓存 前端开发 JavaScript
"面试通关秘籍:深度解析浏览器面试必考问题,从重绘回流到事件委托,让你一举拿下前端 Offer!"
【10月更文挑战第23天】在前端开发面试中,浏览器相关知识是必考内容。本文总结了四个常见问题:浏览器渲染机制、重绘与回流、性能优化及事件委托。通过具体示例和对比分析,帮助求职者更好地理解和准备面试。掌握这些知识点,有助于提升面试表现和实际工作能力。
54 1
|
20天前
|
前端开发 JavaScript UED
"前端小技巧大揭秘:JS如何将后台时间戳秒变亲切小时前、分钟前,让用户秒懂,提升互动体验!"
【10月更文挑战第23天】在Web开发中,将后台返回的时间戳转换为“小时前”、“分钟前”、“刚刚”等友好的时间描述是常见需求。本文介绍如何用JavaScript实现这一功能,通过计算当前时间和时间戳的差值,返回相应的描述,提升用户体验。
25 1
|
8天前
|
前端开发 JavaScript 安全
vite3+vue3 实现前端部署加密混淆 javascript-obfuscator
【11月更文挑战第7天】本文介绍了在 Vite 3 + Vue 3 项目中使用 `javascript-obfuscator` 实现前端代码加密混淆的详细步骤。包括项目准备、安装 `javascript-obfuscator`、配置 Vite 构建以应用混淆,以及最终构建项目进行混淆。通过这些步骤,可以有效提升前端代码的安全性,防止被他人轻易分析和盗用。