關(guān)于JavaScript語句后面的分號問題
JavaScript自動加分號規(guī)則,有3條
- 當(dāng)有換行符(包括含有換行符的多行注釋),并且下一個token沒法跟前面的語法匹配時,會自動補分號。
- 當(dāng)有}時,如果缺少分號,會補分號
- 當(dāng)程序源代碼結(jié)束時,如果缺少分號,會補分號。
利用我自己的JS語法分析工具JSinJS(https://github.com/kissjs/JSinJS ),我求出了所有能夠出現(xiàn)在語句第一個的JS語法標(biāo)記 (就是Statement的first集合),他們是:
["debugger", "try", "throw", "switch", "Identifier", "with", "return", "break", "continue", "for", "while", "do", "if", "new", "function", "(", "{", "[", "RegularExpressionLiteral", "StringLiteral", "NumericLiteral", "BooleanLiteral", "NullLiteral", "this", "!", "~", "-", "+", "--", "++", "typeof", "void", "delete", ";", "var"]
共計35個。
我又求出了所有可以出現(xiàn)在分號之前的語法標(biāo)記(即去掉分號以后的last集),他們是
["--", "++", "IdentifierName", "]", ")", "}", "RegularExpressionLiteral", "StringLiteral", "NumericLiteral", "BooleanLiteral", "NullLiteral", "Identifier", "this", "debugger", "return", "break", "continue"]
共計17個。
35*17 = 595種組合,為了方便記憶,以下我分組來討論語法歧義 。(本來用Excel弄了張表,不過表太大了不好貼出來)
首先,以下語法標(biāo)記開頭的語句是絕對安全的,不會跟不加分號的上一行產(chǎn)生任何歧義:
var if do while for continue break return with switch throw try debugger ;
接下來我們來分組看不加分號導(dǎo)致的語法歧義:
- 第一種是++和--兩種運算符出現(xiàn)在上一行結(jié)尾的情況,下一行以以下開頭時,會產(chǎn)生語法歧義:
- function delete void typeof new null true false NumericLiteral StringLiteral RegularExpressionLiteral ( [ { Identifier ++ -- + - ~ !
- 其中,function和delete是非常常用的statement開頭。
- 特別是 ++和--單獨被斷為一行的時候,因為JS的語法規(guī)則規(guī)定后自增運算不允許中間插入換行,所以++和--會被視為前自增而跟下一行連接在一起。
- 第二種是return作為上一行結(jié)尾的情況,下一行以以下開頭時,會產(chǎn)生語法歧義:
- function delete void typeof ( [ { Identifier ++ -- + - ~ !
- 同樣因為JS語法的規(guī)則不允許在return 和后面的值之間插入換行,所以return之后只要有換行符就會視為有分號,這常常會與使用者的期望不符合。
- 第三種是下一行以+和-開頭的情況,上一行以以下結(jié)尾是,會產(chǎn)生語法歧義:
- -- ++ IdentifierName ] ) } RegularExpressionLiteral
- 因為很少有語句以+或者-開頭,所以這種情況不算危險。
- 第四種是上一行以break、continue結(jié)尾的情況,下一行以Identifier開頭時,會產(chǎn)生語法歧義。
- 第五種是下一行以(和[開頭的情況,上一行以以下結(jié)尾是,會產(chǎn)生語法歧義:
- -- ++ IdentifierName ] ) } RegularExpressionLiteral StringLiteral NumericLiteral BooleanLiteral NullLiteral Identifier this
- 這種情況非常危險(所以hax的文章中要提出這種情況應(yīng)該語句前寫分號),幾乎上一行的所有情況都將導(dǎo)致正常期望之外的結(jié)果。
- 第六種是,當(dāng)下一行以RegularExpressionLiteral 開頭的情況,上一行的以下結(jié)尾,會導(dǎo)致/被理解為除號:
- -- ++ IdentifierName ] ) } RegularExpressionLiteral StringLiteral NumericLiteral BooleanLiteral NullLiteral Identifier this
總結(jié),
- 在return、break、continue、后自增、后自減五種語句中,換行符可以完全替代分號的作用。
- var if do while for continue break return with switch throw try debugger幾種關(guān)鍵字開頭的語句,以及空語句,上一行加不加分號影響不大。
- 凡表達(dá)式語句和函數(shù)表達(dá)式語句,后面不加分號非常危險,情況極其復(fù)雜。
- 凡(和[開頭的語句,前面不加分號極度危險。
下面在通過實例代碼介紹下JavaScript中的分號問題
一般在比較懶的前臺程序員中經(jīng)常會碰到一些莫名其妙的問題。
今天僅討論一下在JS中經(jīng)常會碰到一些關(guān)于分號的問題。JavaScript這門語言是可以省略分號的,是因為它會換行符后如果缺少分號就會無法編譯時它會默認(rèn)添加上分號,但是在一些特定情況下他是不會默認(rèn)添加分號的?,F(xiàn)在簡單介紹一下需要注意的幾個地方。
在這種情況時:
var x = 0 [x+1,x+2,x+3].forEach(function(){ console.log(x) })
這種情況下會導(dǎo)致程序無法正常運行。JavaScript在解析這段代碼是并不會在var x = 0后換行。
在寫代碼時如果以一條語句以 ”(” ,"[" ,"+" ,"-" ,"/" 開始時通常在上一條語句不會默認(rèn)添加分號的。所以在這種情況下盡量保持一下這種寫法,在以這些字符開始時在行首添加一個分號,這樣可以保證在別人更改上面代碼時不加分號也不會影響以下代碼運行。
var x = 0 ;[x+1,x+2,x+3].forEach(function(){ console.log(x) })
還有就是在涉及 return break continue 這種語句時盡量不要換行
return true; JavaScript會解析為 return; true; 在涉及 ++ 和 -- 這一系列運算時 在作為表達(dá)式的前綴或后綴時在換行是會有一定的問題,如下情況: var x = 0; var y = 0; x ++ y
JavaScript會解析為
x;++y; 而不是 x++;y;
雖然在JavaScript這門語言中 “;” 是可以省略不寫的,但是還是建議大家每句代碼后都跟上 “;” 養(yǎng)成這種良好的編碼習(xí)慣,畢竟在大多語言中不帶 “;” 的編碼適不適用的。
總結(jié)
以上所述是小編給大家介紹的JavaScript語句后面的分號問題,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
JavaScript之json_動力節(jié)點Java學(xué)院整理
這篇文章主要介紹了JavaScript之json,JSON它是一種數(shù)據(jù)交換格式。有興趣的可以了解一下2017-06-06用JavaScript來美化HTML的select標(biāo)簽的下拉列表效果
這篇文章主要介紹了用JavaScript來美化HTML的select標(biāo)簽的下拉列表效果的方法,而且在多瀏覽器下的兼容效果也得到提升,需要的朋友可以參考下2015-11-11