`

Javascript的陷阱

阅读更多
条件语句(3个陷阱)
所有的条件语句都必须位于圆括号中。执行语句主体不管是一句还是多句都强烈建议用花括号包围起来,这样能避免很多因修改或嵌套而产生的潜在错误。 if(var1 == var2){//statement}不要犯无意地使用赋值运算符的错误:把第二个参数的值赋给第一个参数。因为它是一个逻辑问题,它将一直返回true且不会报错。 if(var1 = var2){} // 返回true。把var2赋值给var1JavaScript是弱类型,除了在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中的许多问题都来自于变量作用域:要么认为局部变量是全局的,要么用函数中的局部变量覆盖了全局变量。为了避免这些问题,最佳方案是根本没有任何全局变量。但是,如果你有一堆,那么你应该知道这些陷阱。

不用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作用域的问题》。
switch语句Estelle Weyl写了一篇switch statement quirks,其要点是:没有数据类型转换 一个匹配,所有的表达式都将执行直到后面的break或return语句执行 你可以对一个单独语句块使用多个case从句。


undefined ≠ nullnull是一个对象,undefined是一个属性、方法或变量。存在null是因为对象被定义。如果对象没有被定义,而测试它是否是null,但因为没有被定义,它无法测试到,而且会抛出错误。if(myObject !== null  && typeof(myObject) !== 'undefined') {
 //如果myObject是undefined,它不能测试是否为null,而且还会抛出错误
}
if(typeof(myObject) !== 'undefined' && myObject !== null) {
 //处理myObject的代码
}


事件处理陷阱刚接触事件处理时最常见的写法就是类似: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