Javascript中var引發(fā)的問(wèn)題
MelonCard發(fā)布了一篇文章——”how one missing var ruined our launch“(”少寫了一個(gè)var毀了我的網(wǎng)站”),這篇文章是說(shuō)MelonCard用Node.js做后臺(tái),因?yàn)槌隽艘粋€(gè)小高峰——有50-100人注冊(cè),結(jié)果整個(gè)網(wǎng)站都不響應(yīng)了,而且還出現(xiàn)了很多奇怪的問(wèn)題。當(dāng)他們調(diào)查到問(wèn)題的要源的時(shí)候,他們發(fā)現(xiàn)下面的代碼少寫了一個(gè)var。
- app.all(‘/apps/:user_id/status’, function(req, res, next) {
- // …
- initial = extractVariables(req.body);
- });
為什么inital少寫一個(gè)var會(huì)引發(fā)這個(gè)問(wèn)題呢?因?yàn)槿绻悴粚憊ar,這個(gè)局部的變量會(huì)被javascript當(dāng)成全局變量,而這個(gè)變量又是 一個(gè)函數(shù),所以,當(dāng)多用戶并發(fā)的時(shí)候,這個(gè)本應(yīng)該在不同用戶下互不干擾的變量,成了各個(gè)用戶共享的東西。試想,用戶A的數(shù)據(jù)被用戶B覆蓋了,用戶A和B的 數(shù)據(jù)還沒(méi)處理完,結(jié)果被新的C給搞亂了,程序的邏輯自然出現(xiàn)了問(wèn)題。
在stackoverflow.com上有這么一個(gè)貼子說(shuō)明了“有var”和“無(wú)var”的差別:
- // These are both globals
- var foo = 1;
- bar = 2;
- function test()
- {
- var foo = 1; // Local
- bar = 2; // Global
- // Execute an anonymous function
- (function()
- {
- var wibble = 1; // Local
- foo = 2; // Inherits from scope above (creating a closure)
- moo = 3; // Global
- }())
- }
上面這個(gè)示例告訴我們,如果你不用var,那么這個(gè)js引擎會(huì)一層一層地向上找父作用域中的變量,如果找到了,就用,如果找不到了,就會(huì)幫你定義一個(gè)全局的變量。上面這個(gè)例子充分說(shuō)明了這一點(diǎn)。所以,如果你想在當(dāng)前的作用域用聲明變量,你一定要用var。這對(duì)于一些亂寫javascript代碼的程序員要注意了。這里再給大家介紹一個(gè)工具——
JSLint( http://www.jslint.com/ ),一個(gè)JS代碼質(zhì)量的分析工具,我們把上述stackoverflow的代碼copy到JSLint這個(gè)在線工具中,我們可以看到下面的報(bào)告:
這個(gè)報(bào)告說(shuō)明了源碼中的那些變量的情況。