电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

十個(gè)JavaScript中易犯的小錯(cuò)誤,你中了幾槍?zhuān)?/h1>
瀏覽:2日期:2023-11-20 14:45:02
序言

在今天,JavaScript已經(jīng)成為了網(wǎng)頁(yè)編輯的核心。尤其是過(guò)去的幾年,互聯(lián)網(wǎng)見(jiàn)證了在SPA開(kāi)發(fā)、圖形處理、交互等方面大量JS庫(kù)的出現(xiàn)。

如果初次打交道,很多人會(huì)覺(jué)得js很簡(jiǎn)單。確實(shí),對(duì)于很多有經(jīng)驗(yàn)的工程師,或者甚至是初學(xué)者而言,實(shí)現(xiàn)基本的js功能幾乎毫無(wú)障礙。但是JS的真實(shí)功能卻比很多人想象的要更加多樣、復(fù)雜。JavaScript的許多細(xì)節(jié)規(guī)定會(huì)讓你的網(wǎng)頁(yè)出現(xiàn)很多意想不到的bug,搞懂這些bug,對(duì)于成為一位有經(jīng)驗(yàn)的JS開(kāi)發(fā)者很重要。

十個(gè)JavaScript中易犯的小錯(cuò)誤,你中了幾槍?zhuān)? src=

常見(jiàn)錯(cuò)誤一:對(duì)于this關(guān)鍵詞的不正確引用

我曾經(jīng)聽(tīng)一位喜劇演員說(shuō)過(guò):

“我從未在這里,因?yàn)槲也磺宄@里是哪里,是除了那里之外的地方嗎?”

這句話(huà)或多或少地暗喻了在js開(kāi)發(fā)中開(kāi)發(fā)者對(duì)于this關(guān)鍵字的使用誤區(qū)。This指代的是什么?它和日常英語(yǔ)口語(yǔ)中的this是一個(gè)意思嗎?

隨著近些年js編程不斷地復(fù)雜化,功能多樣化,對(duì)于一個(gè)程序結(jié)構(gòu)的內(nèi)部指引、引用也逐漸變多起來(lái)

下面讓我們一起來(lái)看這一段代碼:

Game.prototype.restart = function () { this.clearLocalStorage(); this.timer = setTimeout(function(){ this.clearBoard();}, 0); };

運(yùn)行上面的代碼將會(huì)出現(xiàn)如下錯(cuò)誤:

Uncaught TypeError: undefined is not a function

這是為什么?this的調(diào)用和它所在的環(huán)境密切相關(guān)。之所以會(huì)出現(xiàn)上面的錯(cuò)誤,是因?yàn)楫?dāng)你在調(diào)用 setTimeout()函數(shù)的時(shí)候, 你實(shí)際調(diào)用的是window.setTimeout(). 因此,在 setTimeout() 定義的函數(shù)其實(shí)是在window背景下定義的,而window中并沒(méi)有 clearBoard() 這個(gè)函數(shù)方法。

下面提供兩種解決方案。第一種比較簡(jiǎn)單直接的方法便是,把this存儲(chǔ)到一個(gè)變量當(dāng)中,這樣他就可以在不同的環(huán)境背景中被繼承下來(lái):

Game.prototype.restart = function () { this.clearLocalStorage(); var self = this; this.timer = setTimeout(function(){ self.clearBoard();}, 0); };

第二種方法便是用bind()的方法,不過(guò)這個(gè)相比上一種要復(fù)雜一些,對(duì)于不熟悉bind()的同學(xué)可以在微軟官方查看它的使用方法:https://msdn.microsoft.com/zh-cn/library/ff841995

Game.prototype.restart = function () { this.clearLocalStorage(); this.timer = setTimeout(this.reset.bind(this), 0); }; Game.prototype.reset = function(){ this.clearBoard();};

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

常見(jiàn)錯(cuò)誤二:傳統(tǒng)編程語(yǔ)言的生命周期誤區(qū)

另一種易犯的錯(cuò)誤,便是帶著其他編程語(yǔ)言的思維,認(rèn)為在JS中,也存在生命周期這么一說(shuō)。請(qǐng)看下面的代碼:

for (var i = 0; i < 10; i++) { /* ... */ } console.log(i);

如果你認(rèn)為在運(yùn)行console.log() 時(shí)肯定會(huì)報(bào)出 undefined 錯(cuò)誤,那么你就大錯(cuò)特錯(cuò)了。我會(huì)告訴你其實(shí)它會(huì)返回 10嗎。

當(dāng)然,在許多其他語(yǔ)言當(dāng)中,遇到這樣的代碼,肯定會(huì)報(bào)錯(cuò)。因?yàn)閕明顯已經(jīng)超越了它的生命周期。在for中定義的變量在循環(huán)結(jié)束后,它的生命也就結(jié)束了。但是在js中,i的生命還會(huì)繼續(xù)。這種現(xiàn)象叫做 variable hoisting。

而如果我們想要實(shí)現(xiàn)和其他語(yǔ)言一樣的在特定邏輯模塊中具有生命周期的變量,可以用let關(guān)鍵字。

常見(jiàn)錯(cuò)誤三:內(nèi)存泄露

內(nèi)存泄露在js變成中幾乎是一個(gè)無(wú)法避免的問(wèn)題。如果不是特別細(xì)心的話(huà),在最后的檢查過(guò)程中,肯定會(huì)出現(xiàn)各種內(nèi)存泄露問(wèn)題。下面我們就來(lái)舉例說(shuō)明一下:

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);

如果運(yùn)行上面的代碼,你會(huì)發(fā)現(xiàn)你已經(jīng)造成了大量的內(nèi)存泄露,每秒泄露1M的內(nèi)存,顯然光靠GC(垃圾回收器)是無(wú)法幫助你的了。由上面的代碼來(lái)看,似乎是longstr在每次replaceThing調(diào)用的時(shí)候都沒(méi)有得到回收。這是為什么呢?

每一個(gè)theThing結(jié)構(gòu)都含有一個(gè)longstr結(jié)構(gòu)列表。每一秒當(dāng)我們調(diào)用 replaceThing, 它就會(huì)把當(dāng)前的指向傳遞給 priorThing. 但是到這里我們也會(huì)看到并沒(méi)有什么問(wèn)題,因?yàn)?priorThing 每回也是先解開(kāi)上次函數(shù)的指向才會(huì)接受新的賦值。并且所有的這一切都是發(fā)生在 replaceThing 函數(shù)體當(dāng)中,按常理來(lái)說(shuō)當(dāng)函數(shù)體結(jié)束之后,函數(shù)中的本地變量也將會(huì)被GC回收,也就不會(huì)出現(xiàn)內(nèi)存泄露的問(wèn)題了,但是為什么會(huì)出現(xiàn)上面的錯(cuò)誤呢?

這是因?yàn)閘ongstr的定義是在一個(gè)閉包中進(jìn)行的,而它又被其他的閉包所引用,js規(guī)定,在閉包中引入閉包外部的變量時(shí),當(dāng)閉包結(jié)束時(shí)此對(duì)象無(wú)法被垃圾回收(GC)。關(guān)于在JS中的內(nèi)存泄露問(wèn)題可以查看http://javascript.info/tutorial/memory-leaks#memory-management-in-javascript

常見(jiàn)錯(cuò)誤四:比較運(yùn)算符

JavaScript中一個(gè)比較便捷的地方,便是它可以給每一個(gè)在比較運(yùn)算的結(jié)果變量強(qiáng)行轉(zhuǎn)化成布爾類(lèi)型。但是從另一方面來(lái)考慮,有時(shí)候它也會(huì)為我們帶來(lái)很多不便,下面的這些例子便是一些一直困擾很多程序員的代碼實(shí)例:

console.log(false == '0'); console.log(null == undefined); console.log(" trn" == 0); console.log('' == 0); // And these do too! if ({}) // ... if ([]) // ...

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

console.log(NaN == NaN); // false console.log(NaN === NaN); // false console.log(isNaN(NaN)); // true 常見(jiàn)錯(cuò)誤五:低效的DOM操作

js中的DOM基本操作非常簡(jiǎn)單,但是如何能有效地進(jìn)行這些操作一直是一個(gè)難題。這其中最典型的問(wèn)題便是批量增加DOM元素。增加一個(gè)DOM元素是一步花費(fèi)很大的操作。而批量增加對(duì)系統(tǒng)的花銷(xiāo)更是不菲。一個(gè)比較好的批量增加的辦法便是使用 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元素是一個(gè)非常昂貴的操作。但是如果是先把要添加的元素全部創(chuàng)建出來(lái),再把它們?nèi)刻砑由先ゾ蜁?huì)高效很多。

常見(jiàn)錯(cuò)誤6:在for循環(huán)中的不正確函數(shù)調(diào)用

請(qǐng)大家看以下代碼:

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); }; }

運(yùn)行以上代碼,如果頁(yè)面上有10個(gè)按鈕的話(huà),點(diǎn)擊每一個(gè)按鈕都會(huì)彈出 “This is element #10”! 。這和我們?cè)阮A(yù)期的并不一樣。這是因?yàn)楫?dāng)點(diǎn)擊事件被觸發(fā)的時(shí)候,for循環(huán)早已執(zhí)行完畢,i的值也已經(jīng)從0變成了。

我們可以通過(guò)下面這段代碼來(lái)實(shí)現(xiàn)真正正確的效果:

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); }

在這個(gè)版本的代碼中, makeHandler 在每回循環(huán)的時(shí)候都會(huì)被立即執(zhí)行,把i+1傳遞給變量num。外面的函數(shù)返回里面的函數(shù),而點(diǎn)擊事件函數(shù)便被設(shè)置為里面的函數(shù)。這樣每個(gè)觸發(fā)函數(shù)就都能夠是用正確的i值了。

常見(jiàn)錯(cuò)誤7:原型繼承問(wèn)題

很大一部分的js開(kāi)發(fā)者都不能完全掌握原型的繼承問(wèn)題。下面具一個(gè)例子來(lái)說(shuō)明:

BaseObject = function(name) { if(typeof name !== "undefined") { this.name = name; } else { this.name = 'default' } };

這段代碼看起來(lái)很簡(jiǎn)單。如果你有name值,則使用它。如果沒(méi)有,則使用 ‘default’:

var firstObj = new BaseObject(); var secondObj = new BaseObject('unique'); console.log(firstObj.name); // -> 結(jié)果是'default' console.log(secondObj.name); // -> 結(jié)果是 'unique'

但是如果我們執(zhí)行delete語(yǔ)句呢:

delete secondObj.name;

我們會(huì)得到:

console.log(secondObj.name); // -> 結(jié)果是 'undefined'

但是如果能夠重新回到 ‘default’狀態(tài)不是更好么? 其實(shí)要想達(dá)到這樣的效果很簡(jiǎn)單,如果我們能夠使用原型繼承的話(huà):

BaseObject = function (name) { if(typeof name !== "undefined") { this.name = name; } }; BaseObject.prototype.name = 'default';

在這個(gè)版本中, BaseObject 繼承了原型中的name 屬性, 被設(shè)置為了 'default'.。這時(shí),如果構(gòu)造函數(shù)被調(diào)用時(shí)沒(méi)有參數(shù),則會(huì)自動(dòng)設(shè)置為 default。相同地,如果name 屬性被從BaseObject移出,系統(tǒng)將會(huì)自動(dòng)尋找原型鏈,并且獲得 'default'值:

var thirdObj = new BaseObject('unique'); console.log(thirdObj.name); delete thirdObj.name; console.log(thirdObj.name); // -> 結(jié)果是 'default' 常見(jiàn)錯(cuò)誤8:為實(shí)例方法創(chuàng)建錯(cuò)誤的指引

我們來(lái)看下面一段代碼:

var MyObject = function() {} MyObject.prototype.whoAmI = function() { console.log(this === window ? "window" : "MyObj"); }; var obj = new MyObject();

現(xiàn)在為了方便起見(jiàn),我們新建一個(gè)變量來(lái)指引 whoAmI 方法, 因此我們可以直接用 whoAmI() 而不是更長(zhǎng)的obj.whoAmI():

var whoAmI = obj.whoAmI;

接下來(lái)為了確保一切都如我們所預(yù)測(cè)的進(jìn)行,我們可以將 whoAmI 打印出來(lái):

console.log(whoAmI);

結(jié)果是:

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

沒(méi)有錯(cuò)誤!

但是現(xiàn)在我們來(lái)查看一下兩種引用的方法:

obj.whoAmI(); // 輸出 "MyObj" (as expected) whoAmI(); // 輸出 "window" (uh-oh!)

哪里出錯(cuò)了呢?

原理其實(shí)和上面的第二個(gè)常見(jiàn)錯(cuò)誤一樣,當(dāng)我們執(zhí)行 var whoAmI = obj.whoAmI;的時(shí)候,新的變量 whoAmI 是在全局環(huán)境下定義的。因此它的this 是指window, 而不是obj!

正確的編碼方式應(yīng)該是:

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) 常見(jiàn)錯(cuò)誤9:用字符串作為setTimeout 或者 setInterval的第一個(gè)參數(shù)

首先我們要聲明,用字符串作為這兩個(gè)函數(shù)的第一個(gè)參數(shù)并沒(méi)有什么語(yǔ)法上的錯(cuò)誤。但是其實(shí)這是一個(gè)非常低效的做法。因?yàn)閺南到y(tǒng)的角度來(lái)說(shuō),當(dāng)你用字符串的時(shí)候,它會(huì)被傳進(jìn)構(gòu)造函數(shù),并且重新調(diào)用另一個(gè)函數(shù)。這樣會(huì)拖慢程序的進(jìn)度。

setInterval("logTime()", 1000); setTimeout("logMessage('" + msgValue + "')", 1000);

另一種方法是直接將函數(shù)作為參數(shù)傳遞進(jìn)去:

setInterval(logTime, 1000); setTimeout(function() { logMessage(msgValue); }, 1000); 常見(jiàn)錯(cuò)誤10:忽略 “strict mode”的作用

“strict mode” 是一種更加嚴(yán)格的代碼檢查機(jī)制,并且會(huì)讓你的代碼更加安全。當(dāng)然,不選擇這個(gè)模式并不意味著是一個(gè)錯(cuò)誤,但是使用這個(gè)模式可以確保你的代碼更加準(zhǔn)確無(wú)誤。

下面我們總結(jié)幾條“strict mode”的優(yōu)勢(shì):

1. 讓Debug更加容易:在正常模式下很多錯(cuò)誤都會(huì)被忽視掉,“strict mode”模式會(huì)讓Debug極致更加嚴(yán)謹(jǐn)。

2. 防止默認(rèn)的全局變量:在正常模式下,給一個(gè)為經(jīng)過(guò)聲明的變量命名將會(huì)將這個(gè)變量自動(dòng)設(shè)置為全局變量。在strict模式下,我們?nèi)∠诉@個(gè)默認(rèn)機(jī)制。

3. 取消this的默認(rèn)轉(zhuǎn)換:在正常模式下,給this關(guān)鍵字指引到null或者undefined會(huì)讓它自動(dòng)轉(zhuǎn)換為全局。在strict模式下,我們?nèi)∠诉@個(gè)默認(rèn)機(jī)制。

4. 防止重復(fù)的變量聲明和參數(shù)聲明:在strict模式下進(jìn)行重復(fù)的變量聲明會(huì)被抱錯(cuò),如(e.g., var object = {foo: "bar", foo: "baz"};) 同時(shí),在函數(shù)聲明中重復(fù)使用同一個(gè)參數(shù)名稱(chēng)也會(huì)報(bào)錯(cuò),如 (e.g., function foo(val1, val2, val1){}),

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

6. 當(dāng)遇到無(wú)效的delete指令的事后報(bào)錯(cuò):delete指令不能對(duì)類(lèi)中未有的屬性執(zhí)行,在正常情況下這種情況只是默默地忽視掉,而在strict模式是會(huì)報(bào)錯(cuò)的。

結(jié)語(yǔ)

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

原文地址:http://www.toptal.com/javascript/10-most-common-javascript-mistakes 譯文:http://1ke.co/course/136

標(biāo)簽: JavaScript

主站蜘蛛池模板: 伺服电机_直流伺服_交流伺服_DD马达_拓达官方网站 | PSI渗透压仪,TPS酸度计,美国CHAI PCR仪,渗透压仪厂家_价格,微生物快速检测仪-华泰和合(北京)商贸有限公司 | C形臂_动态平板DR_动态平板胃肠机生产厂家制造商-普爱医疗 | 工控机,嵌入式主板,工业主板,arm主板,图像采集卡,poe网卡,朗锐智科 | 北京网站建设公司_北京网站制作公司_北京网站设计公司-北京爱品特网站建站公司 | 国际学校_国际学校哪个好_国际课程学校-国际学校择校网 | 智能交通网_智能交通系统_ITS_交通监控_卫星导航_智能交通行业 | 别墅图纸超市|别墅设计图纸|农村房屋设计图|农村自建房|别墅设计图纸及效果图大全 | 冷库安装厂家_杭州冷库_保鲜库建设-浙江克冷制冷设备有限公司 | 橡胶接头_橡胶软接头_套管伸缩器_管道伸缩器厂家-巩义市远大供水材料有限公司 | 大通天成企业资质代办_承装修试电力设施许可证_增值电信业务经营许可证_无人机运营合格证_广播电视节目制作许可证 | 润滑脂-高温润滑脂-轴承润滑脂-食品级润滑油-索科润滑油脂厂家 | 泰来华顿液氮罐,美国MVE液氮罐,自增压液氮罐,定制液氮生物容器,进口杜瓦瓶-上海京灿精密机械有限公司 | 合肥活动房_安徽活动板房_集成打包箱房厂家-安徽玉强钢结构集成房屋有限公司 | 安全,主动,被动,柔性,山体滑坡,sns,钢丝绳,边坡,防护网,护栏网,围栏,栏杆,栅栏,厂家 - 护栏网防护网生产厂家 | 镀锌钢格栅_热镀锌格栅板_钢格栅板_热镀锌钢格板-安平县昊泽丝网制品有限公司 | 注塑机-压铸机-塑料注塑机-卧式注塑机-高速注塑机-单缸注塑机厂家-广东联升精密智能装备科技有限公司 | 免联考国际MBA_在职MBA报考条件/科目/排名-MBA信息网 | 武汉天安盾电子设备有限公司 - 安盾安检,武汉安检门,武汉安检机,武汉金属探测器,武汉测温安检门,武汉X光行李安检机,武汉防爆罐,武汉车底安全检查,武汉液体探测仪,武汉安检防爆设备 | 至顶网 | 运动木地板_体育木地板_篮球馆木地板_舞台木地板-实木运动地板厂家 | b2b网站大全,b2b网站排名,找b2b网站就上地球网 | 自动检重秤-动态称重机-重量分选秤-苏州金钻称重设备系统开发有限公司 | 中图网(原中国图书网):网上书店,尾货特色书店,30万种特价书低至2折! | 焊管生产线_焊管机组_轧辊模具_焊管设备_焊管设备厂家_石家庄翔昱机械 | CTP磁天平|小电容测量仪|阴阳极极化_双液系沸点测定仪|dsj电渗实验装置-南京桑力电子设备厂 | 元拓建材集团官方网站| 柴油发电机组_柴油发电机_发电机组价格-江苏凯晨电力设备有限公司 | 流量检测仪-气密性检测装置-密封性试验仪-东莞市奥图自动化科技有限公司 | 污水处理设备维修_污水处理工程改造_机械格栅_过滤设备_气浮设备_刮吸泥机_污泥浓缩罐_污水处理设备_污水处理工程-北京龙泉新禹科技有限公司 | 中红外QCL激光器-其他连续-半导体连续激光器-筱晓光子 | 玻璃钢型材-玻璃钢风管-玻璃钢管道,生产厂家-[江苏欧升玻璃钢制造有限公司] | 广州番禺搬家公司_天河黄埔搬家公司_企业工厂搬迁_日式搬家_广州搬家公司_厚道搬迁搬家公司 | 低噪声电流前置放大器-SR570电流前置放大器-深圳市嘉士达精密仪器有限公司 | 实体店商新零售|微赢|波后|波后合作|微赢集团 | 厦门网站建设_厦门网站设计_小程序开发_网站制作公司【麦格科技】 | 全自动在线分板机_铣刀式在线分板机_曲线分板机_PCB分板机-东莞市亿协自动化设备有限公司 | vr安全体验馆|交通安全|工地安全|禁毒|消防|安全教育体验馆|安全体验教室-贝森德(深圳)科技 | 二手Sciex液质联用仪-岛津气质联用仪-二手安捷伦气质联用仪-上海隐智科学仪器有限公司 | 卡诺亚轻高定官网_卧室系统_整家定制_定制家居_高端定制_全屋定制加盟_定制家具加盟_定制衣柜加盟 | 对辊式破碎机-对辊制砂机-双辊-双齿辊破碎机-巩义市裕顺机械制造有限公司 |