java犯的小錯誤_[Java教程]十個JavaScript中易犯的小錯誤,你中了幾槍?

[Java教程]十個JavaScript中易犯的小錯誤,你中了幾槍?

0 2015-06-01 12:00:19

序言

在今天,JavaScript已經成為了網頁編輯的核心。尤其是過去的幾年,互聯網見證了在SPA開發、圖形處理、交互等方面大量JS庫的出現。

如果初次打交道,很多人會覺得js很簡單。確實,對于很多有經驗的工程師,或者甚至是初學者而言,實現基本的js功能幾乎毫無障礙。但是JS的真實功能卻比很多人想象的要更加多樣、復雜。JavaScript的許多細節規定會讓你的網頁出現很多意想不到的bug,搞懂這些bug,對于成為一位有經驗的JS開發者很重要。

bc91bb04e6e9c61e24c974e4440db8f2.gif

常見錯誤一:對于this關鍵詞的不正確引用

我曾經聽一位喜劇演員說過:“我從未在這里,因為我不清楚這里是哪里,是除了那里之外的地方嗎?”

這句話或多或少地暗喻了在js開發中開發者對于this關鍵字的使用誤區。This指代的是什么?它和日常英語口語中的this是一個意思嗎?

隨著近些年js編程不斷地復雜化,功能多樣化,對于一個程序結構的內部指引、引用也逐漸變多起來

下面讓我們一起來看這一段代碼:Game.prototype.restart = function () { this.clearLocalStorage(); this.timer = setTimeout(function(){ this.clearBoard(); }, 0); };

運行上面的代碼將會出現如下錯誤:Uncaught TypeError: undefined is not a function

這是為什么?this的調用和它所在的環境密切相關。之所以會出現上面的錯誤,是因為當你在調用 setTimeout()函數的時候, 你實際調用的是window.setTimeout(). 因此,在 setTimeout()?定義的函數其實是在window背景下定義的,而window中并沒有 clearBoard()?這個函數方法。

下面提供兩種解決方案。第一種比較簡單直接的方法便是,把this存儲到一個變量當中,這樣他就可以在不同的環境背景中被繼承下來:Game.prototype.restart = function () { this.clearLocalStorage(); var self = this; this.timer = setTimeout(function(){ self.clearBoard();}, 0); };

第二種方法便是用bind()的方法,不過這個相比上一種要復雜一些,對于不熟悉bind()的同學可以在微軟官方查看它的使用方法:https://msdn.microsoft.com/zh-cn/library/ff841995Game.prototype.restart = function () { this.clearLocalStorage(); this.timer = setTimeout(this.reset.bind(this), 0); }; Game.prototype.reset = function(){ this.clearBoard();};

上面的例子中,兩個this均指代的是Game.prototype。

常見錯誤二:傳統編程語言的生命周期誤區

另一種易犯的錯誤,便是帶著其他編程語言的思維,認為在JS中,也存在生命周期這么一說。請看下面的代碼:for (var i = 0; i < 10; i++) { /* ... */ } console.log(i);

如果你認為在運行console.log()?時肯定會報出?undefined?錯誤,那么你就大錯特錯了。我會告訴你其實它會返回 10嗎。

當然,在許多其他語言當中,遇到這樣的代碼,肯定會報錯。因為i明顯已經超越了它的生命周期。在for中定義的變量在循環結束后,它的生命也就結束了。但是在js中,i的生命還會繼續。這種現象叫做?variable hoisting。

而如果我們想要實現和其他語言一樣的在特定邏輯模塊中具有生命周期的變量,可以用let關鍵字。

常見錯誤三:內存泄露

內存泄露在js變成中幾乎是一個無法避免的問題。如果不是特別細心的話,在最后的檢查過程中,肯定會出現各種內存泄露問題。下面我們就來舉例說明一下:var theThing = null; var replaceThing = function () { var priorThing = theThing; var unused = function () { if (priorThing) { console.log("hi"); } }; theThing = { longStr: new Array(1000000).join('*'), // someMethod: function () { console.log(someMessage); } }; }; setInterval(replaceThing, 1000);

如果運行上面的代碼,你會發現你已經造成了大量的內存泄露,每秒泄露1M的內存,顯然光靠GC(垃圾回收器)是無法幫助你的了。由上面的代碼來看,似乎是longstr在每次replaceThing調用的時候都沒有得到回收。這是為什么呢?

每一個theThing結構都含有一個longstr結構列表。每一秒當我們調用 replaceThing, 它就會把當前的指向傳遞給 priorThing. 但是到這里我們也會看到并沒有什么問題,因為 priorThing?每回也是先解開上次函數的指向才會接受新的賦值。并且所有的這一切都是發生在 replaceThing?函數體當中,按常理來說當函數體結束之后,函數中的本地變量也將會被GC回收,也就不會出現內存泄露的問題了,但是為什么會出現上面的錯誤呢?

這是因為longstr的定義是在一個閉包中進行的,而它又被其他的閉包所引用,js規定,在閉包中引入閉包外部的變量時,當閉包結束時此對象無法被垃圾回收(GC)。關于在JS中的內存泄露問題可以查看http://javascript.info/tutorial/memory-leaks#memory-management-in-javascript

常見錯誤四:比較運算符

JavaScript中一個比較便捷的地方,便是它可以給每一個在比較運算的結果變量強行轉化成布爾類型。但是從另一方面來考慮,有時候它也會為我們帶來很多不便,下面的這些例子便是一些一直困擾很多程序員的代碼實例:console.log(false == '0'); console.log(null == undefined); console.log(" \t\r\n" == 0); console.log('' == 0); // And these do too! if ({}) // ... if ([]) // ...

最后兩行的代碼雖然條件判斷為空(經常會被人誤認為轉化為false),但是其實不管是{ }還是[ ]都是一個實體類,而任何的類其實都會轉化為true。就像這些例子所展示的那樣,其實有些類型強制轉化非常模糊。因此很多時候我們更愿意用 ===?和?!==?來替代==?和?!=, 以此來避免發生強制類型轉化。. ===和!==?的用法和之前的==?和?!= 一樣,只不過他們不會發生類型強制轉換。另外需要注意的一點是,當任何值與 NaN?比較的時候,甚至包括他自己,結果都是false。因此我們不能用簡單的比較字符來決定一個值是否為 NaN?。我們可以用內置的 isNaN()?函數來辨別:console.log(NaN == NaN); // false console.log(NaN === NaN); // false console.log(isNaN(NaN)); // true

常見錯誤五:低效的DOM操作

js中的DOM基本操作非常簡單,但是如何能有效地進行這些操作一直是一個難題。這其中最典型的問題便是批量增加DOM元素。增加一個DOM元素是一步花費很大的操作。而批量增加對系統的花銷更是不菲。一個比較好的批量增加的辦法便是使用?document fragments?:var div = document.getElementsByTagName("my_div"); var fragment = document.createDocumentFragment(); for (var e = 0; e < elems.length; e++) { fragment.appendChild(elems[e]); } div.appendChild(fragment.cloneNode(true));

直接添加DOM元素是一個非常昂貴的操作。但是如果是先把要添加的元素全部創建出來,再把它們全部添加上去就會高效很多。

常見錯誤6:在for循環中的不正確函數調用

請大家看以下代碼:var elements = document.getElementsByTagName('input');var n = elements.length; for (var i = 0; i < n; i++) { elements[i].onclick = function() { console.log("This is element #" + i); }; }

運行以上代碼,如果頁面上有10個按鈕的話,點擊每一個按鈕都會彈出 “This is element #10”! 。這和我們原先預期的并不一樣。這是因為當點擊事件被觸發的時候,for循環早已執行完畢,i的值也已經從0變成了。

我們可以通過下面這段代碼來實現真正正確的效果:var elements = document.getElementsByTagName('input'); var n = elements.length; var makeHandler = function(num) { // outer function return function() { console.log("This is element #" + num); }; }; for (var i = 0; i < n; i++) { elements[i].onclick = makeHandler(i+1); }

在這個版本的代碼中, makeHandler?在每回循環的時候都會被立即執行,把i+1傳遞給變量num。外面的函數返回里面的函數,而點擊事件函數便被設置為里面的函數。這樣每個觸發函數就都能夠是用正確的i值了。

常見錯誤7:原型繼承問題

很大一部分的js開發者都不能完全掌握原型的繼承問題。下面具一個例子來說明:BaseObject = function(name) { if(typeof name !== "undefined") { this.name = name; } else { this.name = 'default' } };

這段代碼看起來很簡單。如果你有name值,則使用它。如果沒有,則使用 ‘default’:var firstObj = new BaseObject(); var secondObj = new BaseObject('unique'); console.log(firstObj.name); // -> 結果是'default' console.log(secondObj.name); // -> 結果是 'unique'

但是如果我們執行delete語句呢:delete secondObj.name;

我們會得到:console.log(secondObj.name); // -> 結果是 'undefined'

但是如果能夠重新回到 ‘default’狀態不是更好么? 其實要想達到這樣的效果很簡單,如果我們能夠使用原型繼承的話:BaseObject = function (name) { if(typeof name !== "undefined") { this.name = name; } }; BaseObject.prototype.name = 'default';

在這個版本中,?BaseObject?繼承了原型中的name?屬性, 被設置為了?'default'.。這時,如果構造函數被調用時沒有參數,則會自動設置為 default。相同地,如果name?屬性被從BaseObject移出,系統將會自動尋找原型鏈,并且獲得 'default'值:var thirdObj = new BaseObject('unique'); console.log(thirdObj.name); delete thirdObj.name; console.log(thirdObj.name); // -> 結果是 'default'

常見錯誤8:為實例方法創建錯誤的指引

我們來看下面一段代碼:var MyObject = function() {} MyObject.prototype.whoAmI = function() { console.log(this === window ? "window" : "MyObj"); }; var obj = new MyObject();

現在為了方便起見,我們新建一個變量來指引 whoAmI?方法, 因此我們可以直接用 whoAmI()?而不是更長的obj.whoAmI():var whoAmI = obj.whoAmI;

接下來為了確保一切都如我們所預測的進行,我們可以將 whoAmI?打印出來:console.log(whoAmI);

結果是:function () { console.log(this === window ? "window" : "MyObj"); }

沒有錯誤!

但是現在我們來查看一下兩種引用的方法:obj.whoAmI(); // 輸出 "MyObj" (as expected) whoAmI(); // 輸出 "window" (uh-oh!)

哪里出錯了呢?

原理其實和上面的第二個常見錯誤一樣,當我們執行 var whoAmI = obj.whoAmI;的時候,新的變量 whoAmI?是在全局環境下定義的。因此它的this?是指window,?而不是obj!

正確的編碼方式應該是:var MyObject = function() {} MyObject.prototype.whoAmI = function() { console.log(this === window ? "window" : "MyObj"); }; var obj = new MyObject(); obj.w = obj.whoAmI; // still in the obj namespace obj.whoAmI(); // 輸出 "MyObj" (as expected) obj.w(); // 輸出 "MyObj" (as expected)

常見錯誤9:用字符串作為setTimeout 或者 setInterval的第一個參數

首先我們要聲明,用字符串作為這兩個函數的第一個參數并沒有什么語法上的錯誤。但是其實這是一個非常低效的做法。因為從系統的角度來說,當你用字符串的時候,它會被傳進構造函數,并且重新調用另一個函數。這樣會拖慢程序的進度。setInterval("logTime()", 1000); setTimeout("logMessage('" + msgValue + "')", 1000);

另一種方法是直接將函數作為參數傳遞進去:setInterval(logTime, 1000); setTimeout(function() { logMessage(msgValue); }, 1000);

常見錯誤10:忽略 “strict mode”的作用

“strict mode” 是一種更加嚴格的代碼檢查機制,并且會讓你的代碼更加安全。當然,不選擇這個模式并不意味著是一個錯誤,但是使用這個模式可以確保你的代碼更加準確無誤。

下面我們總結幾條“strict mode”的優勢:

1. 讓Debug更加容易:在正常模式下很多錯誤都會被忽視掉,“strict mode”模式會讓Debug極致更加嚴謹。

2.?防止默認的全局變量:在正常模式下,給一個為經過聲明的變量命名將會將這個變量自動設置為全局變量。在strict模式下,我們取消了這個默認機制。

3.?取消this的默認轉換:在正常模式下,給this關鍵字指引到null或者undefined會讓它自動轉換為全局。在strict模式下,我們取消了這個默認機制。

4.?防止重復的變量聲明和參數聲明:在strict模式下進行重復的變量聲明會被抱錯,如(e.g.,?var object = {foo: "bar", foo: "baz"};)?同時,在函數聲明中重復使用同一個參數名稱也會報錯,如 (e.g.,?function foo(val1, val2, val1){}),

5.?讓eval()函數更加安全。

6.?當遇到無效的delete指令的事后報錯:delete指令不能對類中未有的屬性執行,在正常情況下這種情況只是默默地忽視掉,而在strict模式是會報錯的。

結語

正如和其他的技術語言一樣,你對JavaScript了解的的越深,知道它是如何運作,為什么這樣運作,你才會熟練地掌握并且運用這門語言。相反地,如果你缺少對JS模式的認知的話,你就會碰上很多的問題。了解JS的一些細節上的語法或者功能將會有助于你提高編程的效率,減少變成中遇到的問題。

本文網址:http://www.shaoqun.com/a/119118.html

*特別聲明:以上內容來自于網絡收集,著作權屬原作者所有,如有侵權,請聯系我們:admin@shaoqun.com。

JavaScript

0

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/395713.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/395713.shtml
英文地址,請注明出處:http://en.pswp.cn/news/395713.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Kali滲透測試——利用metasploit攻擊靶機WinXP SP1

搭建滲透測試環境 Kali攻擊機 WinXP SP1 靶機 啟動metasploit 跟windows RPC相關的漏洞 內部提供的漏洞攻擊 靶機winxp sp1網絡配置 查看虛擬機的NAT網段 配置WinXP SP1靶機的IP地址 執行漏洞利用 后漏洞利用&#xff1a;meterpreter> 靶機的信息 進程情況 查看到explorer.e…

創建響應式布局的優秀網格工具集錦《系列五》

在這篇文章中&#xff0c;我們為您呈現了一組優秀的網格工具清單。如果網頁設計和開人員采用了正確的工具集&#xff0c;并基于一個靈活的網格架構&#xff0c;以及能夠把響應圖像應用到到設計之中&#xff0c;那么創建一個具備響應式的網站并不一定是一項艱巨的任務。enjoy! 您…

【iOS - 周總結】開發中遇到的小知識點(2018.12.10-2018.12.15)

1.WKWebview加載html文本圖片過大&#xff0c;沒有自適應屏幕寬高。 在用Webview加載html文本有時候會遇到加載的圖片過大&#xff0c;不能自適應屏幕寬高的問題。那么如何解決這個問題&#xff1f;如何使圖片自適應屏幕&#xff1f;很簡單&#xff0c;只需要加一個js就可以。 …

如何使用Create React App DevOps自動化工作中所有無聊的部分

by James Y Rauhut詹姆士魯豪(James Y Rauhut) 如何使用Create React App DevOps自動化工作中所有無聊的部分 (How I automate all of the boring parts of my job with Create React App DevOps) When you have responsibilities as one of the only designers — and possib…

java 無侵入監控_MyPerf4J 一個高性能、無侵入的Java性能監控和統計工具

MyPerf4J一個針對高并發、低延遲應用設計的高性能且無侵入的實時Java性能監控和統計工具。 受 perf4j 和 TProfiler啟發而來。MyPerf4J具有以下幾個特性&#xff1a;無侵入: 采用JavaAgent方式&#xff0c;對應用程序完全無侵入&#xff0c;無需修改應用代碼高性能: 性能消耗非…

Apple Swift編程語言新手教程

文件夾 1 簡單介紹2 Swift入門3 簡單值4 控制流5 函數與閉包6 對象與類7 枚舉與結構1 簡單介紹 今天凌晨Apple剛剛公布了Swift編程語言&#xff0c;本文從其公布的書籍《The Swift Programming Language》中摘錄和提取而成。希望對各位的iOS&OSX開發有所幫…

javascript 減少回流

減少回流&#xff08;REFLOWS&#xff09; 當瀏覽器重新渲染文檔中的元素時需要 重新計算它們的位置和幾何形狀&#xff0c;我們稱之為回流。回流會阻塞用戶在瀏覽器中的操作&#xff0c;因此理解提升回流時間是非常有幫助的。 回流時間圖表 你應該批量地觸發回流或重繪&#x…

[國家集訓隊] 特技飛行

題目背景 1.wqs愛好模擬飛行。 2.clj開了一家神犇航空&#xff0c;由于clj還要玩游戲&#xff0c;所以公司的事務由你來打理。 注意&#xff1a;題目中只是用了這樣一個背景&#xff0c;并不與真實/模擬飛行相符 題目描述 神犇航空開展了一項載客特技飛行業務。每次飛行長N個單…

react 手指移開_代碼簡介:React的五個死亡手指

react 手指移開Here are three stories we published this week that are worth your time:這是我們本周發布的三個值得您關注的故事&#xff1a; React’s Five Fingers of Death. Master these five concepts, then master React: 10 minute read React的五指死亡。 掌握這五…

java lock接口_Java Lock接口

Java Lock接口java.util.concurrent.locks.Lock接口用作線程同步機制&#xff0c;類似于同步塊。新的鎖定機制更靈活&#xff0c;提供比同步塊更多的選項。 鎖和同步塊之間的主要區別如下&#xff1a;序列的保證 - 同步塊不提供對等待線程進行訪問的序列的任何保證&#xff0c;…

springcloud-05-ribbon中不使用eureka

ribbon在有eureka的情況下, 可以不使用eureka, 挺簡單, 直接上代碼 application.xml server:port: 7002 spring:# 設置eureka中注冊的名稱, 全小寫, 否則大小寫混雜出現問題application:name: microservice-consumer-movie-ribben-ymllogging:level:root: INFOorg.hibernate: I…

SQL mysql優化

慢查詢 如何通過慢查日志發現有問題的SQL&#xff1f; 查詢次數多且每次查詢占用時間長的SQL pt-query-digest分析前幾個查詢IO大的SQL pt-query-diges分析中的Rows examine項未命中索引的SQL pt-query-digest分析中Rows examine 和Rows Send的對比如何分析SQL查詢 使用explain…

轉: 關于 ssl的建立鏈接的過程

轉自&#xff1a; http://www.ruanyifeng.com/blog/2014/02/ssl_tls.html SSL/TLS協議運行機制的概述 作者&#xff1a; 阮一峰 日期&#xff1a; 2014年2月 5日 互聯網的通信安全&#xff0c;建立在SSL/TLS協議之上。 本文簡要介紹SSL/TLS協議的運行機制。文章的重點是設計思…

第一章第一個c#程序上機_我從第一個#100DaysOfCode中學到的東西

第一章第一個c#程序上機On May 17th, I completed my first round of #100DaysOfCode. In case you haven’t heard, #100DaysOfCode is a challenge, or movement, started by Alexander Kallaway for people interested in coding. The basis of the challenge is that you p…

[Swift通天遁地]一、超級工具-(2)制作美觀大方的環形進度條

★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★?微信公眾號&#xff1a;山青詠芝&#xff08;shanqingyongzhi&#xff09;?博客園地址&#xff1a;山青詠芝&#xff08;https://www.cnblogs.com/strengthen/&#xff09;?GitHub地址&a…

SPOJ QTREE6 lct

題目鏈接 島娘出的題。還是比較easy的 #include <iostream> #include <fstream> #include <string> #include <time.h> #include <vector> #include <map> #include <queue> #include <algorithm> #include <stack> #in…

使用charles 抓取手機上的操作

Charles上的設置要截取iPhone上的網絡請求&#xff0c;我們首先需要將Charles的代理功能打開。在Charles的菜單欄上選擇“Proxy”->“Proxy Settings”&#xff0c;填入代理端口8888&#xff0c;并且勾上”Enable transparent HTTP proxying” 就完成了在Charles上的設置。如…

FreeCodeCamp納什維爾聚會的回顧

by Seth Alexander塞斯亞歷山大(Seth Alexander) FreeCodeCamp納什維爾聚會的回顧 (A Recap from the freeCodeCamp Nashville Meetup) At a recent freeCodeCamp meetup, a small group of campers got together to solve some coding challenges and we talk shop.在最近的f…

php查詢車位系統代碼,php車輛違章查詢數據示例

方便有車一族隨時了解自己是否有過交通違章&#xff0c;避免因遺忘或逾期處理違章罰單而造成的不必要損失。本代碼示例是基于聚合數據全國車輛違章查詢API的調用&#xff0c;有需要的可以往下看。使用前你需要&#xff1a;一、引入封裝好的請求類class.juhe.wz.phpheader(Conte…

[HNOI2011]XOR和路徑

嘟嘟嘟 一看到異或&#xff0c;就想到按位處理&#xff0e; 當處理到第\(i\)位的時候&#xff0c;\(f[u]\)表示節點\(u\)到\(n\)的路徑&#xff0c;這一位為\(1\)的期望&#xff0c;那么為\(0\)就是\(1 - f[u]\)&#xff0c;于是有\[f[u] \frac{1}{d[u]} (\sum _ {v \in V, w …