博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
ES6 Module之export
阅读量:6710 次
发布时间:2019-06-25

本文共 4144 字,大约阅读时间需要 13 分钟。

只是搬迁自己之前丢在简书的文字写于初学es6时所以语句不通(虽然并无价值)

是只是export,所以......import忽略

本篇代码运行环境为{"presets": [ "es2015","stage-2" ] }这只是作为参考,且只是运行环境,不推荐在学习ES6时将代码全部转译为ES5,ES6转化后的代码只能告诉你结果,相比较而言,原因或是理由的价值超过结果,学习ES6,不单单要知道代码运行的结果,最重要的目的是,知其所以然,了解一个行为为什么会这么发生,行为的背后又是什么,这才是学习者所需要追寻的

文章参考:

首先登场的就是 export , 这是一个主要关键词,基本用法是放置在一个"声明"之前,或一组由{}语法(注意,此处的{}语法与对象无关)包裹的即将被导出的"标识符"之前

//export 放置在"声明"之前export var a = 1, b = 2, c = 3export let a = 1export const a = 1export let { a } = { a: 1 }export var foo = function() {}export function foo() {}//export 放置在一组"标识符"之前var a = 1, b = 2export { a, b }//等同于export var a = 1export var b = 2复制代码

以上例子有一个明确的共同点,export 后面没有出现“表达式”。实际上,单独的export 是对变量标识符(指针位置)的绑定,并期许将来会把对应的标识符(指针)导出。

将“变量标识符”导出,这样的描述容易产生混淆,考虑下面的代码

var a = 1export { a }a = 3//等同于export var a = 1a = 3复制代码

当这个模块被导出后,如果赋值发生,那么已被导出的值也将被更新,无论导出发生在任何阶段。进一步的说,在被导入时的值是无关紧要的。这些绑定是实时的链接,所以唯一重要的是当你访问这个绑定时它当前的值是什么。

  1. 声明
  2. 导出标识符 a,此刻的 a 是指向变量本身的一个引用,或指针,而不是它的值的一个拷贝
  3. 模块内部,a 被重新赋值,已经导出的值也会被自动更新
参考规范15.2.3.2ExportDeclaration : export VariableStatement  1. Return the BoundNames of VariableStatement.ExportDeclaration : export Declaration  1. Return the BoundNames of Declaration.可以看到export 的导出是明确的复制代码

另外“标识符”一词引用于The above rule means that each ReferencedBindings of ExportClause is treated as an IdentifierReference.主要是词穷,并且因为模块导出与赋值是不同的,在进行导出时,实质上是导出了一个单向绑定的(不允许在导入的一方进行改变)变量的引用,是的,更准确的说应该是对于变量这个容器的引用,模块导出并不关心变量的值。

在进行导出的时候,可以使用别名,关键词 as

var a = 1export { a as b }//将 a 重命名为 b复制代码

"在一个模块中,使用import命令的时候,用户需要知道所要加载的变量名或函数名,否则无法加载。但是,用户肯定希望快速上手,未必愿意阅读文档,去了解模块有哪些属性和方法。"

"为了给用户提供方便,让他们不用阅读文档就能加载模块,就要用到export default命令,为模块指定默认输出。"

上述两句直接摘自,主要是感觉是很恰当的描述,如果加以变动反倒画蛇添足了,当然还不够全面,所以对default进来以下的补充

  1. 每个模块定义只能有一个default,它是唯一的,每个被导出的模块只包含一个default元素,所以export default命令在模块内只被允许使用一次。

  2. 本质上export default就是输出一个叫做default的默认标识符。等同于,将export default 之后的内容以赋值的形式添加到default元素上。

var a = 1export default a //等同于export default 1//导出的是那表达式在那一刻的值的绑定,不是 标识符a的绑定(export.default = 表达式)复制代码

export default 有许多微妙的细节,令人困扰的(不是结果,而是行为)。请思考下面的代码。

//m2.js1.function foo() {}export default foofoo = 'change'2.export default (function foo() {})foo = 'change'3.function foo() {}export { foo as default }foo = 'change'//由于上文已经描述过,default接近于标识符,所以,可以直接重命名foo作为default导出。4.export default function foo() {}foo = 'change'//另一模块import * as all from 'm2.js'console.log(all)复制代码

你的大脑能够清晰的知道每个模块即将会发生的事情吗?如果不能那么请继续阅读,如果能,那么也希望你继续阅读,重温复习这一片段。下面让我复制代码,描述并解释每一模块的行为。 ######模块1.

function foo() {}//声明fooexport default foo//将foo赋值给default元素(注意,此时foo是表达式)foo = 'change'//结果{ default: [Function: foo] }复制代码

export default 导出的是那一个函数表达式在那一刻的值的绑定,不是 标识符foo的绑定。换句话说,export default ..接收一个表达式。如果你稍后在你的模块内部赋给foo一个不同的值,这个模块导入将依然表示原本被导出的函数,而不是那个新的值。

规范里定义了export default 表达式的导出相关行为export default AssignmentExpression

ExportDeclaration : export default AssignmentExpression ; 1.Return «"default"». 简单解释下,就是将表达式的值赋予default,然后返回default ######模块2.

export default (function foo() {})将(..)赋值给default 元素(注意,()是表达式)foo = 'change'//结果ReferenceError: foo is not definedexport default !function foo() {}!等运算符可以包装一个函数使它作为一个表达式返回值复制代码

export default (function foo(){}),export default后面的是函数表达式,并不是函数声明定义,所以它对应的规范与模块1相同,导出的ExportedBindings也就是一个«"default"»。

这里之所以报错是因为函数表达式只会返回函数本身作为值,并不会在外部作用域定义同名变量,所以下面的foo = 'change'找不到foo这个定义。

######模块3.

function foo() {}export { foo as default }foo = 'change'//结果{ default: 'change' }复制代码

ExportDeclaration : export Declaration 1.Return the BoundNames of Declaration. 行为与 export '标识符‘相同,所以引用的规范相同,唯一需要理解的是default是可以被赋值

######模块4.

export default function foo() {}//一个函数声明出现了!foo = 'change'//结果{ default: 'change' }复制代码

function foo..部分是一个函数表达式,但是对于模块内部作用域来说,它被视为一个函数声明,因为名称foo被绑定在模块的顶层作用域

export default 函数声明定义在规范中定义的行为,对应的是export default HoistableDeclaration. ExportDeclaration : export default HoistableDeclaration 1.Let declarationNames be the BoundNames of HoistableDeclaration. 2.If declarationNames does not include the element "default", append "default" to declarationNames. 3.Return declarationNames.

可以看到按照规范如果将要导出的声明没有包含元素default,那么就进行赋值(规范概念),最后返回的是一个当前绑定的标识符,与前面的表达式时状态不同。所以结果能够看到,导出值被更新了。

原标题为ES6 Module 详解,然后发现自己并没有真正理解Module这部分于是停笔,回去看书了,实际意义上本文是为了总结知识

初学ES6,文章有误请指点,文内部分用词不准确也请谅解,虽然引用了规范,但是并没有能力进行解读,惭愧。

文章参考:

转载地址:http://poalo.baihongyu.com/

你可能感兴趣的文章
OpenGL学习之路(三)
查看>>
嵌入式系统 Boot Loader 技术内幕【转】
查看>>
(windows)一台电脑上安装两个Mysql服务
查看>>
教你如何在Kali Linux 环境下设置蜜罐?
查看>>
主域控角色迁移和夺取(转载)
查看>>
HDFS High Availability Using the Quorum Journal Manager
查看>>
Sql日期时间格式转换
查看>>
mesos+marathon+zookeeper的docker管理集群亲手搭建实例(环境Centos6.8)
查看>>
你应了解的4种JS设计模式
查看>>
垃圾收集器Serial 、Parallel、CMS、G1
查看>>
mongodb基本概念解析
查看>>
OpenCV【2】---读取png图片显示到QT label上的问题
查看>>
Azure China (12) 域名备案问题
查看>>
STM32 常用GPIO操作函数记录
查看>>
XML:使用DOM技术解析xML文件中的城市,实现select级联选择
查看>>
JQuery:JQuery添加元素
查看>>
图解GitHub基本操作
查看>>
linux集群时间同步搭建
查看>>
C++12.1.4 类的前向声明、不完全类型类
查看>>
H5以及Node读取excel
查看>>