中文字幕二区_国产精品免费在线观看_黄色网站观看_人人草人人澡_日本真实娇小xxxx

您的位置: 首頁 > 技術(shù)文檔 > 網(wǎng)頁制作 > JavaScript中盡量用局部變量的原因
提升JavaScript運行速度之循環(huán)篇 回到列表 JavaScript的2008
 JavaScript中盡量用局部變量的原因

作者:明達 時間: 2009-02-19 文檔類型:翻譯 來自:七月佑安

在JavaScript中,我們應(yīng)該盡可能的用局部變量來代替全局變量,這句話所有人都知道,可是這句話是誰先說的?為什么要這么做?有什么根據(jù)么?不這么做,對性能到底能帶來多大的損失?本文就來探討這些問題的答案,從根本上了解變量的讀寫性能都和哪些因素有關(guān)。

著作權(quán)聲明

本文譯自 Nicholas C. Zakas 于2009年2月10日在個人網(wǎng)站上發(fā)表的《JavaScript Variable Performance》。原文是唯一的正式版,本文是經(jīng)過原作者(Nicholas C. Zakas)授權(quán)的簡體中文翻譯版(Simplified Chinese Translation)。譯者(明達)在翻譯的準確性上做了大量的努力,并承諾譯文的內(nèi)容完全忠于原文,但可能還是包含疏漏和不妥之處,歡迎大家指正。譯注的內(nèi)容是非正式的,僅代表譯者個人觀點。

以下是對原文的翻譯

在如何提高JavaScript性能這個問題上,大家最常聽到的建議應(yīng)該就是盡量使用局部變量(local variables)來代替全局變量(global variables)。在我從事Web開發(fā)工作的九年時間里,這條建議始終縈繞在我的耳邊,并且從來沒有質(zhì)疑過,而這條建議的基礎(chǔ),則來自于 JavaScript處理作用域(scoping)和標識符解析(identifier resolution)的方法。

首先我們要明確,函數(shù)在JavaScript中具體表現(xiàn)為對象,創(chuàng)建一個函數(shù)的過程,其實也就是創(chuàng)建一個對象的過程。每個函數(shù)對象都有一個叫做 [[Scope]]的內(nèi)部屬性,這個內(nèi)部屬性包含創(chuàng)建函數(shù)時的作用域信息。實際上,[[Scope]]屬性對應(yīng)的是一個對象(Variable Objects)列表,列表中的對象是可以從函數(shù)內(nèi)部訪問的。比如說我們建立一個全局函數(shù)A,那么A的[[Scope]]內(nèi)部屬性中只包含一個全局對象(Global Object),而如果我們在A中創(chuàng)建一個新的函數(shù)B,那么B的[[Scope]]屬性中就包含兩個對象,函數(shù)A的Activation Object對象在前面,全局對象(Global Object)排在后面。

當一個函數(shù)被執(zhí)行的時候,會自動創(chuàng)建一個可以執(zhí)行的對象(Execution Object),并同時綁定一個作用域鏈(Scope Chain)。作用域鏈會通過下面兩個步驟來建立,用于進行標識符解析。

  1. 首先將函數(shù)對象[[Scope]]內(nèi)部屬性中的對象,按順序復制到作用域鏈中。
  2. 其次,在函數(shù)執(zhí)行時,會創(chuàng)建一個新的Activation Object對象,這個對象中包含了this、參數(shù)(arguments)、局部變量(包括命名的參數(shù))的定義,這個Activation Object對象會被置于作用域鏈的最前面。

在執(zhí)行JavaScript代碼的過程中,當遇到一個標識符,就會根據(jù)標識符的名稱,在執(zhí)行上下文(Execution Context)的作用域鏈中進行搜索。從作用域鏈的第一個對象(該函數(shù)的Activation Object對象)開始,如果沒有找到,就搜索作用域鏈中的下一個對象,如此往復,直到找到了標識符的定義。如果在搜索完作用域中的最后一個對象,也就是全局對象(Global Object)以后也沒有找到,則會拋出一個錯誤,提示用戶該變量未定義(undefined)。這是在ECMA-262標準中描述的函數(shù)執(zhí)行模型和標識符解析(Identifier Resolution)的過程,事實證明,大部分的JavaScript引擎確實也是這樣實現(xiàn)的。需要注意的是,ECMA-262并沒有強制要求采用這種結(jié)構(gòu),只是對這部分功能加以描述而已。

了解標識符解析(Identifier Resolution)的過程以后,我們就能明白為什么局部變量的解析速度要比其他作用域的變量快,主要是由于搜索過程被大幅縮短了。但是,具體會快多少呢?為了回答這個問題,我模擬了一系列的測試,來測試不同作用域深度中變量的性能。

第一個測試是向一個變量中寫入一個最簡單的值(這里使用字面量的數(shù)值1),結(jié)果如下圖顯示,很有趣:

從結(jié)果中不難看出,當標識符解析的過程需要進行深度搜索時,會伴隨性能損失,而且性能損失的程度會隨著標識符深度的增加而遞增。意料之中的是,Internet Explorer表現(xiàn)的是最差的(但公平的說,IE 8還是有一些改善的)。值得注意的是,這里有一些例外情況,Google Chrome和最新的WebKit午夜版在訪問變量的時間保持得很穩(wěn)定,不會隨著作用域深度的遞增而增長。當然,這應(yīng)該歸功于它們所使用的下一代 JavaScript引擎,V8和SquirrelFish。這些引擎在執(zhí)行代碼時進行了優(yōu)化,而且很明顯,這些優(yōu)化使訪問變量的速度比以往更快。 Opera表現(xiàn)的也很不錯,比IE、Firefox和當前版本的Safari要快的多,但比基于V8和Squirrelfish的瀏覽器要慢。 Firefox 3.1 Beta 2的表現(xiàn)有點出人意料,對于局部變量執(zhí)行的效率非常高,但隨著作用域?qū)訑?shù)的增加,效率便大打折扣。需要注意的是,我這里使用的都是默認設(shè)置,也就是說 Firefox是沒有開啟Trace功能的。

上面的結(jié)果是通過對變量執(zhí)行寫操作而得出的,其實我很好奇,讀取變量時的情況會不會有什么不同,于是接著做了下面的測試。結(jié)果發(fā)現(xiàn),讀的速度要比寫的速度快一些,但是性能變化的趨勢是一致的。

和上個測試一樣,Internet Explorer和Firefox還是最慢的,Opera表現(xiàn)了非常搶眼的性能,而同樣的,Chrome和最新版本的Webkit午夜版顯示了和作用域深度無關(guān)的性能趨勢,同樣需要注意的是,F(xiàn)irefox 3.1 Beta 2的變量訪問時間還是會伴隨著深度出現(xiàn)一個奇怪的跳躍。

在測試的過程中,我發(fā)現(xiàn)一個有趣的現(xiàn)象,就是Chrome在訪問全局變量的時候會有額外的性能損失。訪問全局變量的時間和作用域?qū)訑?shù)沒有關(guān)系,但是會比訪問同樣層數(shù)的局部變量的時間多出50%。

這兩個測試可以給我們帶來什么啟示呢?首先是驗證了那個古老的觀點,就是要盡可能的使用局部變量。在所有的瀏覽器下,訪問局部變量都比訪問跨作用域的變量要快,當然也包括全局變量。下面這幾點應(yīng)該是通過這個測試得出的經(jīng)驗吧:

  • 仔細檢查函數(shù)中所有使用的變量,如果有一個變量不是當前作用域定義的,而且使用了不止一次,那么我們就應(yīng)該把這個變量保存在局部變量中,而使用這個局部變量來進行讀寫操作。這樣可以幫助我們將作用域外的變量的搜索深度減少到1.這對全局變量尤為重要,因為全局變量總是被放到作用域鏈的最后位置來搜索。
  • 避免使用with語句。因為它會修改執(zhí)行上下文(Execution Context)的作用域鏈,在最前面添加一個對象(Variable Object)。這就意味著在執(zhí)行with的過程中,實際上的局部變量都被移到作用域鏈上的第二個位置,這會帶來性能上的損失。
  • 如果你確定一段代碼肯定會拋出異常,那么就要避免使用try-catch,因為catch分支在作用域鏈上的處理方法和with是一樣的。但try分支的代碼是沒有性能損失的,所以還是建議用try-catch來捕獲那些不可預(yù)知的錯誤。

如果你想圍繞這個話題展開更多的討論,我在上個月的 Mountain View JavaScript Meetup 中曾經(jīng)發(fā)表了一個小演講。可以在SlideShare上下載 幻燈片,或者觀看聚會的 完整視頻,我的演講大概從11分鐘左右時開始。

譯者筆記

大家如果在閱讀本文的過程中,有什么疑惑,建議延伸閱讀以下兩篇文章:

在最后的時候,Nicholas提到一個Mountain View JavaScript Meetup,Meetup那個網(wǎng)站其實就是一個各種現(xiàn)實世界活動的組織網(wǎng)站,需要翻墻才能訪問,住在California真幸福,有那么多的好活動可以參加,呵呵。

本文鏈接:http://m.95time.cn/tech/web/2009/6430.asp 

出處:七月佑安
責任編輯:moby

◎進入論壇網(wǎng)頁制作、WEB標準化版塊參加討論,我還想發(fā)表評論。

相關(guān)文章 更多相關(guān)鏈接
提升JavaScript運行速度之循環(huán)篇
JavaScript的2008
JavaScript Tween算法及緩動效果
仿Apple滑動條(拖動)產(chǎn)品展示效果
如何走出JavaScript初學困境
作者文章
瀏覽器根據(jù)什么來判定腳本失控?
提升JavaScript運行速度之循環(huán)篇
JavaScript的2008
關(guān)鍵字搜索 常規(guī)搜索 推薦文檔
熱門搜索:CSS Fireworks 設(shè)計比賽 網(wǎng)頁制作 web標準 用戶體驗 UE photoshop Dreamweaver Studio8 Flash 手繪 CG
站點最新 站點最新列表
周大!熬•自然”設(shè)計大賽開啟
國際體驗設(shè)計大會7月將在京舉行
中國國防科技信息中心標志征集
云計算如何讓安全問題可控
云計算是多數(shù)企業(yè)唯一擁抱互聯(lián)網(wǎng)的機會
阿里行云
云手機年終巨獻,送禮標配299起
阿里巴巴CTO王堅的"云和互聯(lián)網(wǎng)觀"
1499元買真八核 云OS雙蛋大促
首屆COCO桌面手機主題設(shè)計大賽
欄目最新 欄目最新列表
淺談JavaScript編程語言的編碼規(guī)范
如何在illustrator中繪制臺歷
Ps簡單繪制一個可愛的鉛筆圖標
數(shù)據(jù)同步算法研究
用ps作簡單的作品展示頁面
CSS定位機制之一:普通流
25個最佳最閃亮的Eclipse開發(fā)項目
Illustrator中制作針線縫制文字效果
Photoshop制作印刷凹凸字體
VS2010中創(chuàng)建自定義SQL Rule

藍色理想版權(quán)申明:除部分特別聲明不要轉(zhuǎn)載,或者授權(quán)我站獨家播發(fā)的文章外,大家可以自由轉(zhuǎn)載我站點的原創(chuàng)文章,但原作者和來自我站的鏈接必須保留(非我站原創(chuàng)的,按照原來自一節(jié),自行鏈接)。文章版權(quán)歸我站和作者共有。

轉(zhuǎn)載要求:轉(zhuǎn)載之圖片、文件,鏈接請不要盜鏈到本站,且不準打上各自站點的水印,亦不能抹去我站點水印。

特別注意:本站所提供的攝影照片,插畫,設(shè)計作品,如需使用,請與原作者聯(lián)系,版權(quán)歸原作者所有,文章若有侵犯作者版權(quán),請與我們聯(lián)系,我們將立即刪除修改。

您的評論
用戶名:  口令:
說明:輸入正確的用戶名和密碼才能參與評論。如果您不是本站會員,你可以注冊 為本站會員。
注意:文章中的鏈接、內(nèi)容等需要修改的錯誤,請用報告錯誤,以利文檔及時修改。
不評分 1 2 3 4 5
注意:請不要在評論中含與內(nèi)容無關(guān)的廣告鏈接,違者封ID
請您注意:
·不良評論請用報告管理員,以利管理員及時刪除。
·尊重網(wǎng)上道德,遵守中華人民共和國的各項有關(guān)法律法規(guī)
·承擔一切因您的行為而直接或間接導致的民事或刑事法律責任
·本站評論管理人員有權(quán)保留或刪除其管轄評論中的任意內(nèi)容
·您在本站發(fā)表的作品,本站有權(quán)在網(wǎng)站內(nèi)轉(zhuǎn)載或引用
·參與本評論即表明您已經(jīng)閱讀并接受上述條款
推薦文檔 | 打印文檔 | 評論文檔 | 報告錯誤  
專業(yè)書推薦 更多內(nèi)容
網(wǎng)站可用性測試及優(yōu)化指南
《寫給大家看的色彩書1》
《跟我去香港》
眾妙之門—網(wǎng)站UI 設(shè)計之道
《Flex 4.0 RIA開發(fā)寶典》
《贏在設(shè)計》
犀利開發(fā)—jQuery內(nèi)核詳解與實踐
作品集 更多內(nèi)容

雜⑦雜⑧ Gold NORMANA V2