聊聊 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

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

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

相关推荐
奕辰杰3 小时前
关于npm前端项目编译时栈溢出 Maximum call stack size exceeded的处理方案
前端·npm·node.js
JiaLin_Denny4 小时前
如何在NPM上发布自己的React组件(包)
前端·react.js·npm·npm包·npm发布组件·npm发布包
_Kayo_5 小时前
VUE2 学习笔记14 nextTick、过渡与动画
javascript·笔记·学习
路光.5 小时前
触发事件,按钮loading状态,封装hooks
前端·typescript·vue3hooks
我爱996!5 小时前
SpringMVC——响应
java·服务器·前端
咔咔一顿操作6 小时前
Vue 3 入门教程7 - 状态管理工具 Pinia
前端·javascript·vue.js·vue3
kk爱闹6 小时前
用el-table实现的可编辑的动态表格组件
前端·vue.js
漂流瓶jz7 小时前
JavaScript语法树简介:AST/CST/词法/语法分析/ESTree/生成工具
前端·javascript·编译原理
换日线°7 小时前
css 不错的按钮动画
前端·css·微信小程序