聊聊 JavaScript 的 ASI 机制

官方文档

ASI ,automatic semicolon insertion,自动分号插入机制,它是一种程序解析技术 ,它在 JavaScript 程序的语法分析(parsing)阶段起作用。

根据 ES6 规范,某些 JS 语句需要用;来表示语言结束,而某些情况下;可以从源码中省去且不会报错,此时我们可以认为;被 parser 自动插入到符号流 当中去了(这只是形象上的比喻,并没有真实发生,代表编译器在没有分号的情况下也正确理解了程序员的意图),这种机制称为 ASI

🌟行结束符

CR -- 回车 --> \r

LF -- 换行 --> \n

编辑器不会优先 启用 ASI 机制,在遇到行结束符时,编辑器总是试图将行结束符分隔的语句当成同一条语句来解析(有特例),实在不符合正确语法的情况下就会退而求其次,启用 ASI 机制,将分隔的语句当成两条语句处理。比如:

javascript 复制代码
var a = 1
var b = 2
// 此代码段的符号流为  var a = 1  \n  var b = 2

// parser 从左到右解析这个符号流
var a = 1  var b = 2	// Uncaught SyntaxError:Unexpected token 'var'

// 于是报错了,不甘心的 parser 启用 ASI 再次尝试
var a = 1; var b = 2   // 没有报错,解析通过

需要注意的是 JS 中存在几个"脆弱符号",可能导致在 parser 解析符号流的时候误解程序的意图。

"脆弱符号"有:[(/+-

javascript 复制代码
var num = 5
+new Date() - new Date(2009, 10)
// 此代码段的符号流为 var num = 5  \n  +new Date() - new Date(2009, 10)

// parser 从左到右解析这个符号流
var num = 5 + new Date() - new Date(2009, 10)	// 由于 + 有二义性,所以没有语法错误
// 程序意图被曲解

处理这类情况的办法是,在"脆弱符号"前面显式地加上防御性分号

按理说,ASI 是一种备用选择 ,然而 ECMAScript 中,有几种特殊语句是不允许 行结束符存在的,如果这种特殊语句有行结束符,parser 会优先认为行结束符表示语句结束,这时 ASI 不再是候选机制,而是优先启动

特殊语句:return语句、块语句、空语句、continue语句、break语句、throw语句、箭头函数、yield表达式、后自增/自减表达式。

javascript 复制代码
function a() {
	return
    {};
}
a()	  // undefined

// 本来,parser 解析后应该是这样的
function a() {
	return {};
}
a()	   // {} (empty object)

// 实际输出并不符合预期,因为解析结果实际上是这样的
function a() {
	return;
    {}
    ;
}
a()
// 函数体内的代码被解析成了 return 语句、块语句和空语句三条单独的语句

有备用开启 ASI 的,有优先开启 ASI 的,也有不会开启 ASI 的情况。

那就是for循环语句,for循环语句头部的 3 个条件语句必须 有分号隔开,不然 parser 不会自动添加


写在最后

One day you'll leave this world behind. So live a life you will remember! --- Avicii

我是暮星,一枚有志于在前端领域证道的攻城狮。

优质前端内容持续输出中......,欢迎点赞 + 关注 + 收藏

相关推荐
2301_773643627 分钟前
ceph镜像
前端·javascript·ceph
程序员黑豆28 分钟前
AI全栈开发之Java:什么是JDK
前端·后端·ai编程
To_OC28 分钟前
万字解析《JS语言精粹》之第四章:函数15大核心精髓(JS灵魂核心)
前端·javascript·代码规范
宋拾壹31 分钟前
同时添加多个类目
android·开发语言·javascript
IT知识分享36 分钟前
从零开发在线简繁转换工具:OpenCC 实战、避坑经验与方案选型
javascript·python
mqcode39 分钟前
Vue3 + Element Plus + Vite 企业级后台框架搭建全流程
前端
SL-staff40 分钟前
Web 白板技术架构深度解析:从渲染到协作的选型哲学
前端·架构
川冰ICE43 分钟前
JavaScript实战④|天气查询应用,调用API与异步处理
javascript·css·css3
微扬嘴角43 分钟前
react篇4--setState、LazyLoad和Hooks
前端·javascript·react.js
杨梦馨1 小时前
万级数据表格卡死?Web Worker 一招搞定
前端·javascript·vue.js