`
wyzxzws
  • 浏览: 392460 次
  • 性别: Icon_minigender_1
  • 来自: dazhou
社区版块
存档分类
最新评论

JavaScript的陷阱

阅读更多

转自:http://dancewithnet.com/2008/10/27/javascript-gotchas/#event-handlers

   这本来是翻译Estelle Weyl的《15 JavaScript Gotchas》 , 里面介绍的都是在JavaScript编程实践中平时容易出错或需要注意的地方,并提供避开这些陷阱的方法,总体上讲,就是在认清事物本质的基础样要坚持 好的编程习惯,其实这就是Douglas Crockford很久以前提出的JavaScript风格要素问题了,有些内容直接是相同的,具体请看《Javascript风格要素(1)》《Javascript风格要素(2)》 。在翻译的过程中,我又看到了贤安去年翻译的《JavaScript的9个陷阱及评点》 ,其内容又有些交叉在一起,所以我就在现有翻译的基础上做了一个简单的拼合,并依据自己的理解增加了一些注释和解释。

  1. 区分大小写:变量名、属性和方法全部都区分大小写
  2. 不匹配的引号、圆括号或花括号将抛出错误
  3. 条件语句:3个常见陷阱
  4. 换行:一直用分号结束语句来避免常见的换行问题
  5. 标点法:在对象声明的尾部逗号将导致出错
  6. HTML id 冲突
  7. 变量作用域:全局变量对局部变量
  8. 函数重载:当重载不存在时,覆盖函数
  9. 区分string.replace()函数不是全局的
  10. parseInt应该包含两个参数
  11. “this”和绑定问题
  12. 为参数设置默认值,以免你遗漏它们
  13. for each循环是用于对象而不是数组
  14. switch语句需要点技巧
  15. 如果你要检查null,应该先检查undefined
  16. 时间处理陷阱

区分大小写

变量名和函数名都是区分大小写的。就像配错的引号一样,这些大家都知道。但是,由于错误是不作声的,所以这是一个提醒。为自己选择一个命名规则,并坚持它。而且,记住JavaScript中的原生函数和CSS属性都是骆驼拼写法(camelCase)

getElementById
(’myId’) != getElementByID
(’myId’); //它应该是“Id”而不是“ID”


getElementById(’myId
‘) != getElementById(’myID
‘); // “Id”也不等于“ID”



document.getElementById('myId').style.Color
; //返回 "undefined"








不匹配的引号、圆括号或花括号

避免陷入不匹配的引号、圆括号或花括号陷阱的最好方式是编码时一直同时写出打开和关闭这两个元素符号,然后在其中间加入代码。开始:

var myString = ""; //在输入字符串值之前写入这对引号


function myFunction(){
 if(){//关闭每个打开的括弧


 }
}
//统计所有的左括号和右括号数量,并且确保它们相等


alert(parseInt(var1)*(parseInt(var2)+parseInt(var3))); //关闭每个打开的圆括号








每当你打开一个元素,请关闭它。当你添加了关闭圆括号后,你再把函数的参数放进圆括号中。如果有一串圆括号,统计所有打开的圆括号和所有关闭的圆括号,并且确保这两个数字相等。

条件语句(3个陷阱)

  1. 所有的条件语句都必须位于圆括号中。执行语句主体不管是一句还是多句都强烈建议用花括号包围起来,这样能避免很多因修改或嵌套而产生的潜在错误。
    if(var1 == var2){//statement
    
    }
    
    
    
    
    
    
  2. 不要犯无意地使用赋值运算符的错误:把第二个参数的值赋给第一个参数。因为它是一个逻辑问题,它将一直返回true且不会报错。
    if(var1 = var2){} // 返回true。把var2赋值给var1
    
    
    
    
    
    
    
    
  3. JavaScript是弱类型,除了在switch语句 中。当JavaScript在case比较时,它是非弱类型。
    var myVar = 5;
    if(myVar == '5'){ //返回true,因为JavaScript是弱类型
    
    
     alert("hi");  //这个alert将执行,因为JavaScript通常不在意数据类型
    
    
    }
    switch(myVar){
     case '5':
     alert("hi"); //这个alert将不会执行,因为数据类型不匹配
    
    
    }
    
    
    
    
    
    
    

换行

当心JavaScript中的硬换行。换行被解释为表示行结束的分号。即使在字符串中,如果在引号中包括了一个硬换行,那么你会得到一个解析错误(未结束的字符串)。

var bad  = '<ul id="myId">
  <li>some text</li>
  <li>more text</li>
  </ul>'; // 未结束的字符串错误



var good = '<ul id="myId">' +
 ‘<li>some text</li>‘ +
 ‘<li>more text</li>‘ +
 ‘</ul>’; // 正确








前面讨论过的换行被解释为分号的规则并不适用于控制结构这种情况:条件语句关闭圆括号后的换行并不是给其一个分号。

一直使用分号和圆括号,那么你不会因换行而出错,你的代码易于阅读,且除了那些不使用分号的怪异源码外你会少一些顾虑:所以当移动代码且最终导致两个语句在一行时,你无需担心第一个语句是否正确结束。

多余的逗号

在任何JavaScript对象定义中,最后一个属性决不能以一个逗号结尾。Firefox不会出错,而IE会报语法错误。

var theObj = {
        city : "Boston",
        state : "MA",//IE6和IE7中有“缺少标识符、字符串或数字”的错误,IE8 beta2修正了它


  }





HTML id 冲突

JavaScript DOM绑定(JavaScript DOM bindings) 允许通过HTML id索引。在JavaScript中函数和属性共享同一个名字空间。所以,当在HTML中的一个id和函数或属性有相同的名字时,你会得到难以跟踪的逻辑错误。然而这更多是一个CSS最佳实践 的问题,当你不能解决你的JavaScript问题时,想起它是很重要的。


<ul>
<li id="length">1</li>
<li id="thisLength">2</li>
<li id="thatLength">3</li>
</ul>
<script>
var listitems = document.getElementsByTagName('li');
var liCount = listitems.length; //IE下返回的是<li id="length">1</li>这个节点而不是所有<li的数量



var thisLength = document.getElementById('thisLength');
    thatLength = document.getElementById('thatLength');
    //IE下会出现“对象不支持此属性和方法”的错误,IE8 beta2下首次加载页面会出错,刷新页面则不会


    //在IE中thisLength和thatLength直接表示以其为id值的DOM节点,


    //所以赋值时会出错,当有var声明时,IE会把其当着变量,这个时候就正常了。


</script>






如果你要标记(X)HTML,绝不要使用JavaScript方法或属性名作为id的值。并且,当你写JavaScript时,避免使用 (X)HTML中的id值作为变量名。

变量作用域

JavaScript中的许多问题都来自于变量作用域:要么认为局部变量是全局的,要么用函数中的局部变量覆盖了全局变量。为了避免这些问题,最佳方案是根本没有任何全局变量。但是,如果你有一堆,那么你应该知道这些陷阱。

不用var关键字声明的变量是全局的。记住使用var关键字声明变量,防止变量具有全局作用域。在下面例子中,在函数中声明的变量具有全局变量,因为没有使用var关键字声明:

anonymousFuntion1 = function(){
 globalvar = 'global scope'; //全局声明,因为“var”遗漏了


 return localvar;
}();
alert(globalvar); //弹出“global scope”,因为函数中的变量是全局声明



anonymousFuntion2 = function(){
 var localvar = 'local scope'; //使用“var”局部声明


 return localvar;
}();
alert(localvar); //错误 “localvar未定义”。没有全局定义localvar








作为参数引进到函数的变量名是局部的。如果参数名也是一个全局变量的名字,像参数变量一样有局部作用域,这没有冲突。如果你想在函数中改变一个全局变量,这个函数有一个参数复制于这个全局变量名,记住所有全局变脸都是window对象的属性。

var myscope = "global";
function showScope(myscope){
 return myscope; //局部作用域,即使有一个相同名字的全局变量


}
alert(showScope('local'));

function globalScope(myscope){
 myscope = window.myscope
; //全局作用域


 return myscope;
}
alert(globalScope(’local’));






你甚至可以在循环中声明变量:

for(var i = 0; i < myarray.length; i++){}





覆盖函数/重载函数

当你不止一次的声明一个函数时,这个函数的最后一次声明将覆盖掉该函数的所有前面版本且不会抛出任何错误或警告。这不同于其他的编程语言,像 Java,你能用相同的名字有多重函数,只要它们有不同的参数:调用函数重载。在JavaScript中没有重载。这使得不能在代码中使用 JavaScript核心部分的名字极其重要。也要当心包含的多个JavaScript文件,像一个包含的脚本文件可能覆盖另一个脚本文件中的函数。请使 用匿名函数和名字空间。

(function(){
 // creation of my namespace 创建我的名字空间


 if(!window.MYNAMESPACE) {
  window['MYNAMESPACE'] = {};
 }
 //如果名字空间不存在,就创建它


 //这个函数仅能在匿名函数中访问


 function myFunction(var1, var2){
  //内部的函数代码在这儿


 }
 // 把内部函数连接到名字空间上,使它通过使用名字空间能访问匿名函数的外面 


 window['MYNAMESPACE']['myFunction'] = myFunction;
})(); // 圆括号 = 立即执行


// 包含所有代码的圆括号使函数匿名








这个例子正式为了实现解决上一个陷阱“变量作用域”的最佳方案。匿名函数详细内容请看《Javascript的匿名函数》YUI 整个库只有YAHOO和YAHOO_config两个全局变量,它正是大量应用匿名函数和命名空间的方法来实现,具体请看《Javascript的一种模块模式》

字符串替换

一个常见错误是假设字符串替换方法的行为会对所有可能匹配都产生影响。实际上,JavaScript字符串替换只改变了第一次发生的地方。为了替换所有发生的地方,你需要设置全局标识。同时需要记住String.replace()的第一个参数是一个正则表达式。

var myString = "this is my string";
myString = myString.replace("","%20"); // "this%20is my string"
myString = myString.replace(/ /,"%20"); // "this%20is my string"
myString = myString.replace(/ /g,"%20"); // "this%20is%20my%20string"






parseInt

在JavaScript得到整数的最常见错误是假设parseInt返回的整数是基于10进制的。别忘记第二个参数基数,它能是从2到36之间的任何值。为了确保你不会弄错,请一直包含第二个参数。

parseInt('09/10/08'); //0


parseInt(’09/10/08′,10); //9, 它最可能是你想从一个日期中得到的值








如果parseInt没有提供第二个参数,则前缀为 ‘0x’ 的字符串被当作十六进制,前缀为 ‘0′ 的字符串被当作八进制。所有其它字符串都被当作是十进制的。如果 numString 的前缀不能解释为整数,则返回 NaN(而不是数字)。

‘this’

另一个常见的错误是忘记使用“this”。在JavaScript对象中定义的函数访问这个对象的属性,但没有使用引用标识符“this”。例如,下面是错误的:

function myFunction() {
 var myObject = {
  objProperty: "some text",
  objMethod: function() {
   alert(objProperty);
  }
 };
 myObject.objMethod();
} 

function myFunction() {
 var myObject = {
  objProperty: "some text",
  objMethod: function() {
   alert(this.objProperty);
  }
 };
 myObject.objMethod();
}






有一篇A List Apart文章 用通俗易懂的英文表达了this绑定的问题。

对this使用最大的陷阱是this在使用过程中其引用会发生改变:

<input type="button" value="Gotcha!" id="MyButton">
<script>
var MyObject = function () {
    this.alertMessage = "Javascript rules";
    this.ClickHandler = function() {
        alert(this.alertMessage );

        //返回结果不是”JavaScript rules”,执行MyObject.ClickHandler时,


        //this的引用实际上指向的是document.getElementById("theText")的引用


  }
}();
document.getElementById(”theText”).onclick =  MyObject.ClickHandler
</script>






其解决方案是:


var MyObject = function () {
    var self = this;

    this.alertMessage = “Javascript rules”;
    this.OnClick = function() {
        alert(self.value);
    }
}();






类似问题的更多细节和解决方案请看《JavaScript作用域的问题》

遗漏的参数

当给函数增加一个参数时,一个常见的错误是忘记更新这个函数的所有调用。如果你需要在已经被调用的函数中增加一个参数来处理一个特殊情况下的调用,请给这个函数中的这个参数设置默认值,以防万一在众多脚本中的众多调用中的一个忘记更新。

function addressFunction(address, city, state, country){
 country = country || “US”
; //如果没有传入country,假设 “US”


 span>//剩下代码
}






你也能通过获取arguments来解决。但是在这篇文章我们的注意力在陷阱上。同时在《Javascript风格要素(2)》 也介绍了||巧妙应用。

for关键字

在JavaScript中关键字for有两种使用方式,一个是for语句,一个是for/in语句。for/in语句将遍历所有的对象属性 (attribute),包括方法和属性(property)。决不能使用for/in来遍历数组:仅在当需要遍历对象属性和方法时才使用for/in。

  1. for(var myVar in myObject)语句用一个指定变量无任何规律地遍历对象的所有属性。如果for/in循环的主体删除了一个还没有枚举出的属性,那么该属性就不在枚 举。如果循环主体定义了新属性,那么循环是否枚举该属性则是由JavaScript的实现决定。
  2. for(var 1=0; i < myArray.length; i++)语句会遍历完一个数组的所有元素。

为了解决这个问题,大体上你可以对对象使用 for … in,对数组使用for循环:

listItems = document.getElementsByTagName('li');
for (var listitem in listItems){
 //这里将遍历这个对象的所有属性和方法,包括原生的方法和属性,但不遍历这个数组:出错了!


}
//因为你要循环的是数组对象,所用for循环


for ( var i = 0; i < listItems.length; i++) {
 //这是真正你想要的
}








对象的有些属性以相同的方式标记成只读的、永久的或不可列举的,这些属性for/in无法枚举。实际上,for/in循环
会遍历所有对象的所有可能属性,包括函数和原型中的属性。所有修改原型属性可能对for/in循环带来致命的危害,所以需要采用hasOwnProperty和typeof做一些必要的过滤,最好是用for来代替for/in。

switch语句

Estelle Weyl写了一篇switch statement quirks ,其要点是:

  1. 没有数据类型转换
  2. 一个匹配,所有的表达式都将执行直到后面的break或return语句执行
  3. 你可以对一个单独语句块使用多个case从句

undefined ≠ null

null是一个对象,undefined是一个属性、方法或变量。存在null是因为对象被定义。如果对象没有被定义,而测试它是否是null,但因为没有被定义,它无法测试到,而且会抛出错误。

if(myObject !== null  && typeof(myObject) !== 'undefined') {
 //如果myObject是undefined,它不能测试是否为null,而且还会抛出错误


}
if(typeof(myObject) !== 'undefined' && myObject !== null) {
 //处理myObject的代码


}






Harish Mallipeddi 对undefined和null有一个说明。

事件处理陷阱

刚接触事件处理时最常见的写法就是类似:

window.onclick = MyOnClickMethod





这种做法不仅非常容易出现后面的window.onclick事件覆盖掉前面的事件,还可能导致大名顶顶的IE内存泄露问题。为了解决类似问题,4年前Simon Willison 就写出了很流行的addLoadEvent():

function addLoadEvent(func) {
 var oldonload = window.onload;
 if (typeof window.onload != 'function') {
  window.onload = func;
 }else {
  window.onload = function() {
   oldonload();
   unc();
  }
 }
}
addEvent(window,'load',func1,false);
addEvent(window,'load',func2,false);
addEvent(window,'load',func3,false);






当然在JavaScript库盛行的现在,使用封装好的事件处理机制是一个很好的选择,比如在YUI中就可以这样写:

YAHOO.util.Event.addListener(window, "click", MyOnClickMethod);





分享到:
评论

相关推荐

    Javascript中的陷阱大集合【译】

    本文将详细探讨一些常见的JavaScript陷阱,并提供相应的解决方案。 #### 双等号(==)与类型强制转换 在JavaScript中,双等号`==`操作符在比较两个值时会进行类型转换,这意味着即使两边的数据类型不同,也会尝试...

    javascript-gotchas:javascript 陷阱

    本文将深入探讨一些常见的JavaScript陷阱,并提供相应的解决策略。 1. **变量作用域**: - 全局变量和局部变量的区分不明确可能导致变量污染。全局变量在整个代码中都可以访问,而局部变量仅在函数内部有效。不...

    帮助避免错误的Javascript陷阱清单

    以下是一些常见的JavaScript陷阱及其详细解释: 1. **浮点运算误差**: JavaScript中的浮点数运算可能会导致看似不准确的结果,例如`0.02 / 0.1`并不等于0.2,而是`0.19999999999999998`。这是由于二进制浮点数...

    js-pitfall-examples:常见的 JavaScript 陷阱示例

    在“js-pitfall-examples”这个项目中,我们收集了一些常见的JavaScript陷阱,通过实例来帮助开发者更好地理解并避免这些问题。 1. **变量提升(Hoisting)** 在JavaScript中,变量声明会被提升到它们所在的作用域...

    javascript陷阱 一不小心你就中招了(字符运算)

    ### JavaScript陷阱:字符运算 在JavaScript编程中,字符运算经常被用到,但由于其灵活性以及动态类型特性,很容易出现一些意想不到的结果,导致程序出现错误或不符合预期的行为。本文将详细介绍几种常见的...

    ist的matlab代码-js-gotchas:JavaScript陷阱和常见错误

    JavaScript陷阱和常见错误 1. 0是虚假的,但"0"是正确的。 if ( 0 ) { console . log ( 'yey.' ) ; } else { console . log ( 'boom' ) ; // this will be printed } if ( "0" ) { console . log ( 'yey.' ) ; // ...

    JavaScript修炼之道英文清晰PDF版(含代码)

    此外,书中还会涉及错误处理、调试技巧、性能优化以及如何避免常见的JavaScript陷阱。这些都是实践中必不可少的知识,能帮助你写出更加健壮的代码。 附带的源代码.zip文件包含书中所有示例和练习的源代码,读者可以...

    [JavaScript][PDF][英文版]Oh My JS

    ***mon JavaScript "Gotchas"(常见的JavaScript陷阱):这些陷阱通常是指那些容易出错的地方,比如变量作用域、异步编程的复杂性等,理解这些可以避免在开发中犯一些常见的错误。 7. Asynchronous JS: Callbacks, ...

    Javascript 陷阱 window全局对象

    在探讨JavaScript中的全局对象陷阱时,主要的焦点是理解JavaScript函数和方法中的this关键字的指向问题。在JavaScript中,this关键字的值取决于函数的调用方式,这与Java等其他语言存在明显的不同。在Java中,方法中...

    JavaScript面试题阿里巴巴JavaScript面试题 阿里巴巴

    它的一些特点包括:简洁的语法、避免常见的JavaScript陷阱、自动管理的数组索引等。 #### 3. MVVM 双向绑定实现 MVVM(Model-View-ViewModel)模式通过观察者模式来实现视图和模型之间的双向数据绑定。具体来说,...

    JavaScript程序设计-javascript中的this.pdf

    JavaScript中的`this`关键字是一个非常重要的...理解`this`的关键在于识别函数是如何被调用的,这对于避免常见的JavaScript陷阱至关重要。在编写JavaScript代码时,正确地使用`this`可以提高代码的可读性和可维护性。

    javascript中文手册和基本教程

    你将学会如何使用console.log进行调试,如何组织代码结构,以及如何避免常见的JavaScript陷阱。 此外,手册可能还会涉及JavaScript的最新特性,如ES6引入的箭头函数、模板字符串、解构赋值、类和模块等,以及后续...

    超实用的javascript代码段30道题目答案.pdf

    【JavaScript基础概念与常见陷阱】 JavaScript是一种广泛应用于Web开发的脚本语言,它在浏览器环境中执行,赋予网页动态功能。...理解这些问题有助于开发者避免常见的JavaScript陷阱并写出更健壮的代码。

    JavaScript基础练习_day1

    这是一个常见的JavaScript陷阱,因为实际上`null`不是对象,但它在`typeof`操作中的结果是`object`。这是由于历史遗留问题导致的。 以上就是从题目中提取出的一些JavaScript基础知识点的详细介绍。对于初学者来说,...

    悟透javascript

    - 常见错误分析:理解并避免常见的JavaScript陷阱,如变量提升、类型转换错误等。 - 性能优化:减少DOM操作,使用事件委托,以及缓存策略来提升页面性能。 - 兼容性问题:处理不同浏览器对JavaScript支持的差异,...

    11-this:从JavaScript执行上下文的视角讲清楚this_For_vip_user_0011

    在JavaScript中,`this`是一个特殊的关键词,它用于在函数或方法中引用当前上下文的对象。`this`的值取决于函数被调用的方式,而不是函数声明的位置,这与...正确理解和使用`this`可以避免许多常见的JavaScript陷阱。

    JavaScript的9个陷阱及评点分析

    JavaScript是一种广泛应用于网页和网络应用开发的脚本语言,它具有独特的语法特性和陷阱,对于初学者或经验丰富的开发者来说,...了解并避免这些JavaScript陷阱,有助于编写更健壮、可维护的代码,让编程生活更加轻松。

Global site tag (gtag.js) - Google Analytics