这个问题在不少文章中都讨论过,在 ESlint 规范中也由于加不加分号而分为两大阵营,到于加不加分号,关键是须要了解分号对于 JavaScript 的影响,开始以前能够先看看下面这道面试题:javascript
请问这段代码是否可以正常运行?html
var a = 1 (function() { console.log(2) })()
。前端
。java
。程序员
。面试
若是运行这段代码,会出现下面的错误:segmentfault
Uncaught TypeError: 1 is not a function
什么鬼!1 is not a function
?咱们没有打算运行数字 1,为何说数字 1 不是函数,这种错误是很难找到缘由的,常常会在出问题的代码行上打转。这个错误必然是上吗的代码在运行时被看做是同一行,其概念以下:后端
var a = 1(function() { /* */ })()
所以当即函数的 ()
被附加在 1 上,这是一个调用函数的语法,因此会产生 1 is not a function
的错误,想要避免这个错误就须要使用分号:服务器
var a = 1 // 随便把分号放在哪里,只要能隔开就行 ;(function() { console.log(2) })()
ASI是 “Automatic Semicolon Insertion” 的缩写,在运行时会往有些折行的代码中自动插入分号,这个机制可使部分代码在没有加入分号时也能正常运行,好比下面的例子:微信
var b = 1 ++b console.log('b', b)
因为代码中的 ++
属于一元表达式,它只能在表达式的左边或右边放置变量,若是没有 ASI 的机制,代码会被转换为 var b = 1 ++ b
这样的错误语句。不过好在有 ASI,在实际运行时会自动被加入分号,也就不会出现上面的错误。
var b = 1; ++b; console.log('b', b); // 2
再来看一个例子,下面的代码在 return
的后面空一行后再写要返回的值,那么问运行结果是什么呢?
function fn() { return '小明' } console.log(fn())
这段程序代码由于 ASI 的修正,return
的后面会被加上一个分号,因此 return
与预期返回的值被分开了,结果 return
的内容为空值,最终的结果也只能是 undefined
。
function fn() { return; '小明'; } console.log(fn()); // undefined
原本 ASI 是出于一片好心,用来修正没有加入分号的代码片断,但恰恰在有的地方并无发挥它的做用(例如本文一开始所介绍的当即函数),致使代码出现了错误;甚至有些代码不会出错,但会使你的代码执行结果和预期相差万里。
解决 ASI 问题的方式以下:
下面时各类不会自动加入分号的规则:
(
、[
、/
字符开始的,这类状况通常会直接出现 Uncaught TypeError
从而致使代码没法运行。var a = 1 var b = a (a + b).toString() var a = 1 [1,2,3].forEach(bar) (function() { })() (function() { })() var a = 1 var b = a /test/.test(b)
+
,-
,*
,%
开始,这类状况大多会影响运算结果,因此应该合并为一行。var a = 2 var b = a +a
,
或 .
开始,这种用法常常会出现,主要是为了不代码过长而加入的分隔,这种状况并不会影响运行,若是善用的话会使代码更容易阅读。var a = 2 var b = a .toString() console.log(typeof b) var a = 1 ,b = 2 // b 一样会被 var 声明
若是遇到须要加入分号的状况,除了能够在语句的末尾加入分号外,也能够把分号加在“不会自动加入分号”的最前方,例如 ()
自己不会自动加入分号,在有这种需求时能够将 ;
加到前面(ESLint Standard JS 规范就用这个方法避免错误)。
// 运行错误 (function() { })() (function() { })() // 正确 ;(function() { })() ;(function() { })()
有的人认为不加分号可让代码看起来更干净和精简,并且在大部分状况下并不会出现错误,因此不少人在敲代码时不会加分号。
不过我更倾向于更严格的规范,也许是由于我是从后端转到前端的,习惯了。至于到底怎么选,只要搞清楚运行上的限制,无论哪一种风格都是挺不错的,只要你喜欢就好。