开发者最容易犯的13个JavaScript错误

分类:Javascript| 发布:camnprbubuol| 查看: | 发表时间:2011/4/19

开发者最容易犯的JavaScript错误,总结出13个。这些当中可能少不了你犯的错误。我们描述了这些陋习,并列出来解决办法,希望对开发者有帮助。

1.for...数组迭代的用法 Usage of for..in to iterate Arrays

举例:

  1. var myArray = [ “a”, “b”, “c” ];  
  2. var totalElements = myArray.length;  
  3. for (var i = 0; i < totalElements; i++) {  
  4. console.log(myArray[i]);  
  5. }  

这里主要的问题是语句中的“for..."不能保证顺序,这意味着你将获得不同的执行结果。此外,如果有人增加一些其他自定义功能的函数Array.prototype,你的循环将重复遍历这些函数,就像原数组项。

解决办法:一直使用规则的for循环来遍历数组。

  1. var myArray = [ “a”, “b”, “c” ];  
  2. for (var i=0; i<myArray.length; i++) {  
  3. console.log(myArray[i]);  
  4. }  

2.数组维度Array dimensions

举例

  1. var myArray = new Array(10); 

第二个问题是开发者使用数组构成器来创建数组,技术上是正确的,然而会比文字符号(literal notation)慢解决办法:使用文字符号来初始化数组,不要预定义数组长度。

var myArray = [];

3.未定义属性 Undefined properties

举例:

  1. var myObject = {  
  2.     someProperty: “value”,  
  3.     someOtherProperty: undefined  

未定义属性,将在对象中创建元素(键‘someOtherProperty’和值‘undefined’.)。如果你遍历数组,检测已存在的元素,那么下面的语句将都返回“未定义/undefined”

  1. typeof myObject[’someOtherProperty’] // undefined  
  2.  
  3. typeof myObject[’unknownProperty’] // undefined  

解决办法:如果你想明确声明对象中的未初始化的属性,标记它们为Null(空)。

  1. var myObject = {  
  2. someProperty: “value”,  
  3. someOtherProperty: null  

4.闭包的滥用Misuse of Closures

举例:

  1. function(a, b, c) {  
  2. var d = 10;  
  3. var element = document.getElementById(‘myID’);  
  4. element.onclick = (function(a, b, c, d) {  
  5. return function() {  
  6. alert (a + b + c + d);  
  7. }  
  8. })(a, b, c, d);  
  9. }  

这里开发者使用两个函数来传递参数a、b、c到onclick handler。双函数根本不需要,徒增代码的复杂性。

变量abc已经在局部函数中被定义,因为他们已经在主函数中作为参数被声明。局部函数中的任何函数都可创建主函数中定义的所有变量的闭包。因此不需要再次传递它们。

看看这里JavaScript Closures FAQ了解更多。

解决办法:使用闭环来简化你的代码。

  1. function (a, b, c) {  
  2. var d = 10;  
  3. var element = document.getElementById(‘myID’);  
  4. element.onclick = function() {  
  5. //a, b, and c come from the outer function arguments.  
  6. //d come from the outer function variable declarations.  
  7. //and all of them are in my closure  
  8. alert (a + b + c + d);  
  9. };  
  10. }  

5.循环中的闭包Closures in loops

举例:

  1. var elements = document.getElementByTagName(‘div’);  
  2. for (var i = 0; i<elements.length; i++) {  
  3. elements[i].onclick = function() {  
  4. alert(“Div number “ + i);  
  5. }  
  6. }  

在这里例子里面,当用户点击不同的divs时,我们想触发一个动作(显示“Div number 1”, “Div number 2”… 等) 。然而,如果你在页面有10个divs,他们全部都会显示“Div number 10”。

问题是当我们使用局部函数创建一个闭包时,函数中的代码可以访问变量i。关键是函数内部i和函数外部i涉及同样的变量。当我们的循环结束,i指向了值10,所以局部函数中的i的值将是10。

解决办法:使用第二函数来传递正确的值。

  1. var elements = document.getElementsByTagName(‘div’);  
  2. for (var i = 0; i<elements.length; i++) {  
  3. elements[i].onclick = (function(idx) { //Outer function  
  4. return function() { //Inner function  
  5. alert(“Div number “ + idx);  
  6. }  
  7. })(i);  
  8. }  

6.DOM对象的内测泄漏Memory leaks with DOM objects

举例:

  1. function attachEvents() {  
  2. var element = document.getElementById(‘myID’);  
  3. element.onclick = function() {  
  4. alert(“Element clicked”);  
  5. }  
  6. };  
  7. attachEvents();  

该代码创建了一个引用循环。变量元素包含函数的引用(归于onclick属性)。同时,函数保持一个DOM元素的引用(提示函数内部可以访问元素,因为闭包。)。所以JavaScript垃圾收集器不能清除元素或是函数,因为他们被相互引用。大部分的JavaScript引擎对于清除循环应用都不够聪明。

解决办法:避免那些闭包,或者不去做函数内的循环引用。

  1. function attachEvents() {  
  2. var element = document.getElementById(‘myID’);  
  3. element.onclick = function() {  
  4. //Remove element, so function can be collected by GC  
  5. delete element;  
  6. alert(“Element clicked”);  
  7. }  
  8. };  
  9. attachEvents();  

7.区别整数数字和浮点数字Differentiate float numbers from integer numbers

举例:

  1. var myNumber = 3.5;  
  2. var myResult = 3.5 + 1.0; //We use .0 to keep the result as float  

在JavaScript中,浮点与整数间没有区别。事实上,JavaScript中的每个数字都表示使用双精度64位格式IEEE 754。简单理解,所有数字都是浮点。

解决办法:不要使用小数(decimals),转换数字(numbers)到浮点(floats)。

  1. var myNumber = 3.5;  
  2. var myResult = 3.5 + 1; //Result is 4.5, as expected  

8.with()作为快捷方式的用法Usage of with() as a shortcut

举例:

  1. team.attackers.myWarrior = { attack: 1, speed: 3, magic: 5};  
  2. with (team.attackers.myWarrior){  
  3. console.log ( “Your warrior power is ” + (attack * speed));  
  4. }  

讨论with()之前,要明白JavaScript contexts如何工作的。每个函数都有一个执行context(语句),简单来说,包括函数可以访问的所有的变量。因此context包含arguments和定义变量。

with()真正是做什么?是插入对象到context链,它在当前context和父级context间植入。就像你看到的with()的快捷方式会非常慢。

解决办法:不要使用with() for shortcuts,仅for context injection,如果你确实需要时。

  1. team.attackers.myWarrior = { attack: 1, speed: 3, magic: 5};  
  2. var sc = team.attackers.myWarrior;  
  3. console.log(“Your warrior power is ” + (sc.attack * sc.speed));  

9.setTimeout/setInterval 字符串的用法 Usage of strings with setTimeout/setInterval

举例:

  1. function log1() { console.log(document.location); }  
  2. function log2(arg) { console.log(arg); }  
  3. var myValue = “test”;  
  4. setTimeout(“log1()”, 100);  
  5. <\/script>')
    365据说看到好文章不转的人,服务器容易宕机
    原创文章如转载,请注明:转载自郑州网建-前端开发 http://camnpr.com/
    本文链接:http://camnpr.com/archives/269.html