淺析Javascript的自動分號插入(ASI)機制
前言
相信從事過C#和Java的大家都知道分號是用作斷句(EOS,end of statement)的,而且必須加分號,否則編譯就不通過了。但JavaScript由于存在ASI機制,因此允許我們省略分號。ASI機制不是說在解析過程中解析器自動把分號添加到代碼中,而是說解析器除了分號還會以換行為基礎(chǔ)按一定的規(guī)則作為斷句的依據(jù),從而保證解析的正確性。
規(guī)范理論
es5 標(biāo)準(zhǔn)定義了自動分號插入規(guī)則,包括以下三個基本規(guī)則加兩個前置條件:
前置條件
1、如果插入分號后解析結(jié)果是空語句,那么不會自動插入分號。
例子:(空語句,else 前不加分好)
if (a > b) else c = d
2、如果插入分號后它成為 for 語句頭部的兩個分號之一,那么不會自動插入分號。
例子:(不會加分號)
for (a; b )
基本規(guī)則
左到右解析程序,當(dāng)遇到一個不符合任何文法產(chǎn)生式的 token(叫做 違規(guī) token(offending token)),那么只要滿足下面條件之一就在違規(guī) token 前面自動插入分號。
1、至少一個 LineTerminator 分割了違規(guī) token 和前一個 token。
2、違規(guī) token 是 }。
例子:(1、2不符合任何產(chǎn)生式,并且之間存在 LineTerminator,因此在違規(guī) token 2前加了分好,2和}則是因為違規(guī) token 是 }所以加了分號)
{ 1
2 } 3
{ 1
;2 ;} 3;
左到右解析程序,tokens 輸入流已經(jīng)結(jié)束,當(dāng)解析器無法將輸入 token 流解析成單個完整 ECMAScript 程序 ,那么就在輸入流的結(jié)束位置自動插入分號。
對于受限產(chǎn)生式,也就是下面的5個,我們把產(chǎn)生式 [no LineTerminator here]后面的 token 叫做受限 token,如果在 token 和 受限 token 間存在了至少一個 LineTerminator,那么會在受限 token 前自動加上 token。
受限的產(chǎn)生式只限如下5個:
PostfixExpression :
LeftHandSideExpression [no LineTerminator here] ++ LeftHandSideExpression [no LineTerminator here] --
ContinueStatement :
continue [no LineTerminator here] Identifier;
BreakStatement :
break [no LineTerminator here] Identifier;
ReturnStatement :
return [no LineTerminator here] Expression;
ThrowStatement : throw [no LineTerminator here] Expression;
歸納
避免 ASI 帶來的問題
1、后綴運算符 ++ 或 -- 和它的操作數(shù)應(yīng)該出現(xiàn)在同一行。
2、return 或 throw 語句的表達式開始位置應(yīng)該和 return 或 throw token 同一行。
3、break 或 continue 語句的標(biāo)示符應(yīng)該和 break 或 continue token 同一行。
何時加分號
無分號黨(懶人黨)想要不加分號,那么就需要知道什么時候應(yīng)該要加分號。網(wǎng)上的一篇文章歸納了 NO ASI 并且會出現(xiàn)錯誤的幾種情況,在這幾種情況下我們是要加分號的。下面是對應(yīng)的描述:
在以 ([/+- 開頭的語句前加分號(由于正常寫法均不會出現(xiàn)以 .,*% 作為語句開頭,因此只需記住前面5個即可,你看能懶則懶哦)
不過這里只考慮了換行的情況,其實 ASI 還存在不換行的情況,這就要根據(jù)標(biāo)準(zhǔn)里的三條規(guī)則行事了!
知道了這點,其實我們就可以省略大部分的分號了。但是也不強求,因為這還是要根據(jù)個人習(xí)慣以及團隊風(fēng)格走的。
小補充
為什么自執(zhí)行函數(shù)前要加分號?
主要是應(yīng)對代碼合并壓縮時,由于缺少分號;帶來的錯誤。知道了上面的規(guī)則,在 ( 開頭的行前加分號就可以避免錯誤了。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。
相關(guān)文章
微信公眾號網(wǎng)頁授權(quán)登錄的超簡單實現(xiàn)步驟
微信開放平臺為第三方移動程序提供分享、傳播的接口,使用戶可將第三方程序的內(nèi)容發(fā)布給好友或分享至朋友圈,下面這篇文章主要給大家介紹了關(guān)于微信公眾號網(wǎng)頁授權(quán)登錄的超簡單實現(xiàn)方法,需要的朋友可以參考下2022-06-06
cookie在javascript中的使用技巧以及隱私在服務(wù)器端的設(shè)置
cookie在javascript中的使用技巧,需要的朋友可以參考下2012-12-12

