开发者社区> 行者武松> 正文

十个JavaScript中易犯的小错误,你中了几枪?

简介:
+关注继续查看

在今天,JavaScript已经成为了网页编辑的核心。尤其是过去的几年,互联网见-证了在SPA开发、图形处理、交互等方面大量JS库的出现。

如果初次打交道,很多人会觉得js很简单。确实,对于很多有经验的工程师,或者甚至是初学者而言,实现基本的js功能几乎 毫无障碍。但是JS的真实功能却比很多人想象的要更加多样、复杂。JavaScript的许多细节规定会让你的网页出现很多意想不到的bug,搞懂这些 bug,对于成为一位有经验的JS开发者很重要。

十个JavaScript中易犯的小错误,你中了几枪?

常见错误一:对于this关键词的不正确引用

我曾经听一位喜剧演员说过:

“我从未在这里,因为我不清楚这里是哪里,是除了那里之外的地方吗?”

这句话或多或少地暗喻了在js开发中开发者对于this关键字的使用误区。This指代的是什么?它和日常英语口语中的this是一个意思吗?

随着近些年js编程不断地复杂化,功能多样化,对于一个程序结构的内部指引、引用也逐渐变多起来

下面让我们一起来看这一段代码:


  1. Game.prototype.restart = function () { this.clearLocalStorage(); 
  2.  
  3. this.timer = setTimeout(function(){ this.clearBoard(); }, 0); 
  4.  
  5. }; 

运行上面的代码将会出现如下错误:


  1. Uncaught TypeError: undefined is not a function 

这 是为什么?this的调用和它所在的环境密切相关。之所以会出现上面的错误,是因为当你在调用 setTimeout()函数的时候, 你实际调用的是window.setTimeout(). 因此,在 setTimeout() 定义的函数其实是在window背景下定义的,而window中并没有 clearBoard() 这个函数方法。

下面提供两种解决方案。第一种比较简单直接的方法便是,把this存储到一个变量当中,这样他就可以在不同的环境背景中被继承下来:


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

第二种方法便是用bind()的方法,不过这个相比上一种要复杂一些,对于不熟悉bind()的同学可以在微软官方查看它的使用方法:


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

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

常见错误二:传统编程语言的生命周期误区

另一种易犯的错误,便是带着其他编程语言的思维,认为在JS中,也存在生命周期这么一说。请看下面的代码:


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

如果你认为在运行console.log() 时肯定会报出 undefined 错误,那么你就大错特错了。我会告诉你其实它会返回 10吗。

当然,在许多其他语言当中,遇到这样的代码,肯定会报错。因为i明显已经超越了它的生命周期。在for中定义的变量在循环结束后,它的生命也就结束了。但是在js中,i的生命还会继续。这种现象叫做 variable hoisting。

而如果我们想要实现和其他语言一样的在特定逻辑模块中具有生命周期的变量,可以用let关键字。

常见错误三:内存泄露

内存泄露在js变成中几乎是一个无法避免的问题。如果不是特别细心的话,在最后的检查过程中,肯定会出现各种内存泄露问题。下面我们就来举例说明一下:


  1. var theThing = null
  2. var replaceThing = function () { 
  3. var priorThing = theThing; 
  4. var unused = function () {  
  5. if (priorThing) { console.log("hi"); }  
  6. }; 
  7.   
  8. theThing = { longStr: new Array(1000000).join('*'), //  
  9. someMethod: function () { console.log(someMessage); }  
  10. }; 
  11. }; 
  12. setInterval(replaceThing, 1000); 

如果运行上面的代码,你会发现你已经造成了大量的内存泄露,每秒泄露1M的内存,显然光靠GC(垃圾回-收器)是无法帮助你的了。由上面的代码来看,似乎是longstr在每次replaceThing调用的时候都没有得到回-收。这是为什么呢?

每一个theThing结构都含有一个longstr结构列表。每一秒当我们调用 replaceThing, 它就会把当前的指向传递给 priorThing. 但是到这里我们也会看到并没有什么问题,因为 priorThing 每回也是先解开上次函数的指向才会接受新的赋值。并且所有的这一切都是发生在 replaceThing 函数体当中,按常理来说当函数体结束之后,函数中的本地变量也将会被GC回-收,也就不会出现内存泄露的问题了,但是为什么会出现上面的错误呢?

这是因为longstr的定义是在一个闭包中进行的,而它又被其他的闭包所引用,js规定,在闭包中引入闭包外部的变量时,当闭包结束时此对象无法被垃圾回-收(GC)。

常见错误四:比较运算符

JavaScript中一个比较便捷的地方,便是它可以给每一个在比较运算的结果变量强行转化成布尔类型。但是从另一方面来考虑,有时候它也会为我们带来很多不便,下面的这些例子便是一些一直困扰很多程序员的代码实例:


  1. console.log(false == '0'); 
  2. console.log(null == undefined); 
  3. console.log(" \t\r\n" == 0); 
  4. console.log('' == 0); // And these do too! 
  5. if ({}) // ... 
  6. if ([]) // ... 

最 后两行的代码虽然条件判断为空(经常会被人误认为转化为false),但是其实不管是{ }还是[ ]都是一个实体类,而任何的类其实都会转化为true。就像这些例子所展示的那样,其实有些类型强制转化非常模糊。因此很多时候我们更愿意用 === 和 !== 来替代== 和 !=, 以此来避免发生强制类型转化。. ===和!== 的用法和之前的== 和 != 一样,只不过他们不会发生类型强制转换。另外需要注意的一点是,当任何值与 NaN 比较的时候,甚至包括他自己,结果都是false。因此我们不能用简单的比较字符来决定一个值是否为 NaN 。我们可以用内置的 isNaN() 函数来辨别:


  1. console.log(NaN == NaN); // false 
  2. console.log(NaN === NaN); // false 
  3. console.log(isNaN(NaN)); // true 

常见错误五:低效的DOM操作

js中的DOM基本操作非常简单,但是如何能有效地进行这些操作一直是一个难题。这其中最典型的问题便是批量增加DOM元 素。增加一个DOM元素是一步花费很大的操作。而批量增加对系统的花销更是不菲。一个比较好的批量增加的办法便是使用 document fragments :


  1. var div = document.getElementsByTagName("my_div"); 
  2. var fragment = document.createDocumentFragment(); 
  3. for (var e = 0; e < elems.length; e++) { fragment.appendChild(elems[e]); } div.appendChild(fragment.cloneNode(true)); 

直接添加DOM元素是一个非常昂贵的操作。但是如果是先把要添加的元素全部创建出来,再把它们全部添加上去就会高效很多。

常见错误6:在for循环中的不正确函数调用

请大家看以下代码:


  1. var elements = document.getElementsByTagName('input'); 
  2. var n = elements.length; 
  3. for (var i = 0; i < n; i++) { 
  4. elements[i].onclick = function() { 
  5. console.log("This is element #" + i); }; } 

运行以上代码,如果页面上有10个按钮的话,点击每一个按钮都会弹出 “This is element #10”! 。这和我们原先预期的并不一样。这是因为当点击事件被触发的时候,for循环早已执行完毕,i的值也已经从0变成了。

我们可以通过下面这段代码来实现真正正确的效果:


  1. var elements = document.getElementsByTagName('input'); 
  2. var n = elements.length; 
  3. var makeHandler = function(num) { // outer function 
  4. return function() { 
  5. console.log("This is element #" + num); }; }; 
  6. for (var i = 0; i < n; i++) 
  7. { elements[i].onclick = makeHandler(i+1); } 

在这个版本的代码中, makeHandler 在每回循环的时候都会被立即执行,把i+1传递给变量num。外面的函数返回里面的函数,而点击事件函数便被设置为里面的函数。这样每个触发函数就都能够是用正确的i值了。

常见错误7:原型继承问题

很大一部分的js开发者都不能完全掌握原型的继承问题。下面具一个例子来说明:


  1. BaseObject = function(name) { 
  2. if(typeof name !== "undefined")  
  3. this.name = name; } 
  4. else 
  5. this.name = 'default' } }; 

这段代码看起来很简单。如果你有name值,则使用它。如果没有,则使用 ‘default’:


  1. var firstObj = new BaseObject();   
  2. var secondObj = new BaseObject('unique');  
  3. console.log(firstObj.name); // -> 结果是'default' 
  4. console.log(secondObj.name); // -> 结果是 'unique' 

但是如果我们执行delete语句呢:


  1. delete secondObj.name; 

但是如果能够重新回到 ‘default’状态不是更好么? 其实要想达到这样的效果很简单,如果我们能够使用原型继承的话:


  1. BaseObject = function (name) 
  2. if(typeof name !== "undefined"
  3. this.name = name; } }; 
  4. BaseObject.prototype.name = 'default'

在 这个版本中, BaseObject 继承了原型中的name 属性, 被设置为了 'default'.。这时,如果构造函数被调用时没有参数,则会自动设置为 default。相同地,如果name 属性被从BaseObject移出,系统将会自动寻找原型链,并且获得 'default'值:


  1. var thirdObj = new BaseObject('unique'); 
  2. console.log(thirdObj.name);  
  3. delete thirdObj.name; 
  4. console.log(thirdObj.name); // -> 结果是 'default' 

常见错误8:为实例方法创建错误的指引

我们来看下面一段代码:


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

现在为了方便起见,我们新建一个变量来指引 whoAmI 方法, 因此我们可以直接用 whoAmI() 而不是更长的obj.whoAmI():


  1. var whoAmI = obj.whoAmI; 

接下来为了确保一切都如我们所预测的进行,我们可以将 whoAmI 打印出来:


  1. console.log(whoAmI); 

结果是:


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

没有错误!

但是现在我们来查看一下两种引用的方法:


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

哪里出错了呢?

原理其实和上面的第二个常见错误一样,当我们执行 var whoAmI = obj.whoAmI;的时候,新的变量 whoAmI 是在全局环境下定义的。因此它的this 是指window, 而不是obj!

正确的编码方式应该是:


  1. var MyObject = function() {} 
  2. MyObject.prototype.whoAmI = function() { 
  3. console.log(this === window ? "window" : "MyObj"); }; 
  4. var obj = new MyObject(); 
  5. obj.w = obj.whoAmI; // still in the obj namespace obj.whoAmI(); // 输出 "MyObj" (as expected) 
  6. obj.w(); // 输出 "MyObj" (as expected) 

常见错误9:用字符串作为setTimeout 或者 setInterval的第一个参数

首先我们要声明,用字符串作为这两个函数的第一个参数并没有什么语法上的错误。但是其实这是一个非常低效的做法。因为从系统的角度来说,当你用字符串的时候,它会被传进构造函数,并且重新调用另一个函数。这样会拖慢程序的进度。


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

另一种方法是直接将函数作为参数传递进去:


  1. setInterval(logTime, 1000); 
  2. setTimeout(function() { 
  3. 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的一些细节上的语法或者功能将会有助于你提高编程的效率,减 少变成中遇到的问题。


来源:51CTO

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
运行水晶报表报crdb_adoplus.dll不存在错误的解决方案
转自博客: http://www.cnblogs.com/wingkin/archive/2012/08/22/2651304.html   在VS2010上编译水晶报表提示找提示以下错误: Could not load fi...
807 0
前端常见bug系列4: JavaScript中忘记类型转换所导致的条件判断错误举例
### 一、忘记类型转换的小数比较错误 举个例子,我们要进行一个字符串型的小数的比较:`'8.8'>'8.7'`,结果自然是true。 那么`'8.8'>'8.10'`呢?结果仍然是true,但愿你没有将它误以为是`8.8>8.10`。 ### 二、忘记类型转换的版本号比较错误 类似上面的问题,在进行版本号比较时,一样存在。比如,我们这么比较`'8.8.1'>'8.10.
1687 0
阿里云ECS云服务器初始化设置教程方法
阿里云ECS云服务器初始化是指将云服务器系统恢复到最初状态的过程,阿里云的服务器初始化是通过更换系统盘来实现的,是免费的,阿里云百科网分享服务器初始化教程: 服务器初始化教程方法 本文的服务器初始化是指将ECS云服务器系统恢复到最初状态,服务器中的数据也会被清空,所以初始化之前一定要先备份好。
13865 0
阿里云服务器怎么设置密码?怎么停机?怎么重启服务器?
如果在创建实例时没有设置密码,或者密码丢失,您可以在控制台上重新设置实例的登录密码。本文仅描述如何在 ECS 管理控制台上修改实例登录密码。
20694 0
阿里云服务器端口号设置
阿里云服务器初级使用者可能面临的问题之一. 使用tomcat或者其他服务器软件设置端口号后,比如 一些不是默认的, mysql的 3306, mssql的1433,有时候打不开网页, 原因是没有在ecs安全组去设置这个端口号. 解决: 点击ecs下网络和安全下的安全组 在弹出的安全组中,如果没有就新建安全组,然后点击配置规则 最后如上图点击添加...或快速创建.   have fun!  将编程看作是一门艺术,而不单单是个技术。
18996 0
Fundebug:JavaScript插件支持错误采样
Fundebug的付费套餐主要是根据错误事件数制定的,这是因为每一个发送到我们服务器的事件,都会消耗一定的CPU、内存、磁盘以及带宽资源,尤其当错误事件数非常大时,会对我们的计算资源造成很大压力。 如果您希望采样收集错误,比如“只收集30%的错误”,可以将sampleRate属性设为0.3。
528 0
+关注
行者武松
杀人者,打虎武松也。
17141
文章
2569
问答
文章排行榜
最热
最新
相关电子书
更多
JS零基础入门教程(上册)
立即下载
性能优化方法论
立即下载
手把手学习日志服务SLS,云启实验室实战指南
立即下载