小編給大家分享一下Javascript分號規(guī)則是什么,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!
目前創(chuàng)新互聯(lián)公司已為千余家的企業(yè)提供了網(wǎng)站建設(shè)、域名、網(wǎng)頁空間、網(wǎng)站托管運營、企業(yè)網(wǎng)站設(shè)計、廊坊網(wǎng)站維護(hù)等服務(wù),公司將堅持客戶導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長,共同發(fā)展。分號允許的場景
分號一般允許出現(xiàn)在大部分語句(statement)的末尾,比如 do-while statement , var statements, expression statements , continue , return , break statement, throw, debugger 等
栗子:
do Statement while ( Expression ) ; 4+4; f(); debugger;
僅有一個分號 ; 可以表示空語句——在JS中合法,比如 ;;;
可解析為三個空語句(empty statement)
空語句可用于輔助產(chǎn)生語法合法的解析結(jié)果,如:
while(1);
如果沒有末尾的分號,將會產(chǎn)生解析錯誤 —— 條件循環(huán)后必須跟隨一個語句
分號還會出現(xiàn)在 for 循環(huán) for ( Expression ; Expression ; Expression ) Statement 中
最后,分號還會出現(xiàn)在 字符串 或 正則表達(dá)式中 —— 表示分號本身
分號可以省略的場景
有些場景下分號可以省略,解析器在解析語句時會根據(jù)需要自動插入分號,大概流程可以這樣理解:
書寫省略 => 解析器解析時發(fā)現(xiàn)缺少時會無法正確解析 => 自動添加分號
so 需要明確能自動插入分號的場景,并明確不會自動插入分號且會引起解析錯誤的情況
規(guī)則1:當(dāng)下一個 token (offending token) 和當(dāng)前解析的 token (previous token) 無法組成合法語句,且滿足以下一個或多個條件時,將會在 offending token 前插入一個分號:
還要考慮一種優(yōu)先級更高的條件:如果插入的分號會被解析為一個空語句,或是 for 語句的頭部兩個分號之一,這時不會插入分號(除了 do-while 語句的終止分號外)
規(guī)則2:當(dāng)解析到達(dá)源代碼文件 (input stream) 的末尾時,將自動添加一個分號標(biāo)識解析結(jié)束
規(guī)則3:符合 restricted production 語法的語句 —— 比較難翻譯,看不懂的可以直接看栗子,這種情況主要描述的是:不應(yīng)該出現(xiàn)換行符的地方出現(xiàn)換行符導(dǎo)致插入分號引起原語句含義變化
同時滿足以下條件,將在 offending token 前自動插入一個分號:
其中 restricted production 包括且只有以下:
UpdateExpression[Yield, Await]: LeftHandSideExpression[?Yield, ?Await] [no LineTerminator here] ++ LeftHandSideExpression[?Yield, ?Await] [no LineTerminator here] -- ContinueStatement[Yield, Await]: continue; continue [no LineTerminator here] LabelIdentifier[?Yield, ?Await]; BreakStatement[Yield, Await]: break; break [no LineTerminator here] LabelIdentifier[?Yield, ?Await]; ReturnStatement[Yield, Await]: return; return [no LineTerminator here] Expression [+In, ?Yield, ?Await]; ThrowStatement[Yield, Await]: throw [no LineTerminator here] Expression [+In, ?Yield, ?Await]; ArrowFunction[In, Yield, Await]: ArrowParameters[?Yield, ?Await] [no LineTerminator here] => ConciseBody[?In] YieldExpression[In, Await]: yield [no LineTerminator here] * AssignmentExpression[?In, +Yield, ?Await] yield [no LineTerminator here] AssignmentExpression[?In, +Yield, ?Await]
簡單總結(jié):
使用 a++ 語句時,變量和 ++ 必須在同一行,否則會在 ++ 前插入分號導(dǎo)致語義不同
return throw yield continue break 后如果緊跟著換行,將會自動添加分號
箭頭函數(shù)的 => 之前不應(yīng)該有換行符
栗子 & 可能不符合預(yù)期的情況
符合預(yù)期情況
// 相當(dāng)于 42;"hello" 42 "hello" // offending token 是 } if(x){y()} // previous token 是 ) 且插入分號是 do while 語句的結(jié)束 var a = 1 do {a++} while(a<100) console.log(a) // 不會解析成 b++ 因為 b和++之間存在換行符,會在 b 之后自動插入分號 a = b ++c
可能不符合預(yù)期的情況
const hey = 'hey' const you = 'hey' const heyYou = hey + ' ' + you ['h', 'e', 'y'].forEach((letter) => console.log(letter))
會收到錯誤 Uncaught TypeError: Cannot read property 'forEach' of undefined , 因為 you 和 ['h', 'e', 'y'] 的連接能命中合法語法,故它們之間不會自動插入分號 —— 與預(yù)期不一致,JS嘗試將代碼解析為:
const hey = 'hey'; const you = 'hey'; const heyYou = hey + ' ' + you['h', 'e', 'y'].forEach((letter) => console.log(letter))
再看一種情況:
const a = 1 const b = 2 const c = a + b (a + b).toString()
會引發(fā) TypeError: b is not a function
報錯,因為會被解釋為:
const a = 1 const b = 2 const c = a + b(a + b).toString()
除了 do while 語句外,不會有插入分號作為空語句的其他情況,或作為 for 語句頭部的兩個必要分號 :
if (a > b) else c = d for (a; b )
以上均不是合法的 JS 語句,并且會引起報錯
故以下栗子中的每一個分號都不能省略!!
// for循環(huán)沒有循環(huán)體的情況,每一個分號都不能省略 for (node=getNode(); node.parent; node=node.parent) ;
再看一個帶詳細(xì)注釋的例子:
var // 這一行不會插入分號 ,因為 下一行的代碼不會破壞當(dāng)前行的代碼 a = 1 // 這一行會插入分號 let b = 2 // 再比如這種情況,你的原意可能是定義 `a` 變量,再執(zhí)行 `(a + 3).toString()`, // 但是其實 JavaScript 解析器解析成了,`var a = 2(a + 3).toString()`, // 這時會拋出錯誤 Uncaught TypeError: 2 is not a function var a = 2 (a + 3).toString() // 同理,下面的代碼會被解釋為 `a = b(function(){...})()` a = b (function(){ ... })()
以上都是未能命中規(guī)則1而未插入分號導(dǎo)致解析與預(yù)期不符合的情況
看一個基于規(guī)則3的例子:
(() => { return { color: 'white' } })()
預(yù)期是返回一個包含 color 屬性的對象,但事實上 return 后會被插入一個分號,而導(dǎo)致最終返回 undefined,可以通過在 return 后立刻放置花括號 { :
(() => { return { color: 'white' } })()
省略分號的最佳實踐
不要使用以下單個字符 ( [ / + - 開始一行 , 會極有可能和上一行語句合在一起被解析( ++ 和 -- 不符合單個 +、- 字符)
注意 return break throw continue 語句,如果需要跟隨參數(shù)或表達(dá)式,把它添加到和這些語句同一行,針對 return 返回內(nèi)容較多的情況 (大對象,柯里化調(diào)用,多行字符串等),可以參考規(guī)則1,避免命中該規(guī)則而引起非預(yù)期的分號插入,比如:
return obj.method('abc') .method('xyz') .method('pqr') return "a long string\n" + "continued across\n" + "several lines" totalArea = rect_a.height * rect_a.width + rect_b.height * rect_b.width + circ.radius * circ.radius * Math.PI
后綴運算符 ++ -- 需要和操作變量在同一行使用
當(dāng)然大部分工程化情況下,我們最終會配合Eslint使用帶分號或省略分號規(guī)范
以上是“Javascript分號規(guī)則是什么”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國服務(wù)器、虛擬主機、免備案服務(wù)器”等云主機租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務(wù)可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場景需求。