为JS扩展Array.prototype.indexOf引发的问题及解决办法


Posted in Javascript onJanuary 21, 2015

Array没有indexOf方法,这样在一个数组中查找某个元素的索引时比较麻烦,为了调用方便,于是通过prototype原型扩展了Array.prototype.indexOf(),这样用起来就比较方便了。但是这个自定义的indexOf在对数组进行遍历的时候却出现了问题。

Array.prototype.indexOf = function(item) {  

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

if (this[i] == item)  

return i;  

}  

return -1;  

}

用的时候直接

var arr=[1,2,3,4,5];  

var index=arr.indexOf(1); //index==0

扩展了以后,用起来很爽很方便,一片和谐景象...

但是某次是遍历数组元素的时候,使用for..in..循环,引发了其他的问题,打破了这个和谐的氛围。

var a=["张飞","关羽","刘备","吕布"];  

for(var p in a){  

  document.write(p+"="+a[p]+"<br/>");  

}

本来想输出这四个人的名字,结果输出的是什么呢?

输出的居然是:

//0=张飞  

//1=关羽  

//2=刘备  

//3=吕布  

//indexOf=function(item) { for (var i = 0; i < this.length; i++) { if (this[i] == item) return i; } return -1; } 

除了把名字打出来以外,还额外输出了自己扩展的方法indexOf,但是令人疯狂的是,firefox却是“正常”的,只有四个人的人名,为什么会这样?

输出indexOf,自己扩展的,可以理解,毕竟for..in是遍历一个对象的所有用户定义的属性或者一个数组的所有元素。

那么firefox为什么不会?

后来查了资料才明白,

Array在javascript1.6版本已经支持Array.indexOf(),而我用的firefox是3.5版本,已经支持javascript1.8了,indexOf是其Array本身固有的方法了。

而IE,即使我用的是IE8,也才支持到javascript1.3版本。

所以IE8认为indexOf是“用户定义的属性”,而firefox认为是自己原生支持的固有的属性。

真的是这样吗?

做个实验,把indexOf更名为myIndexOf,再试试,结果IE和firefox都输出myIndexOf,证明前面的观点是正确。

那么又来了个问题,我扩展indexOf很久了,现在不少项目的代码都已经在使用这个方法,而现在我非要使用for..in输出数组本身的元素,不要其他我自己扩展到俄方法,怎么办?

好在javascript提供了hasOwnProperty方法。

看一下其描述:

Every object descended from Object inherits the hasOwnProperty method. This method can be used to determine whether an object has the specified property as a direct property of that object; unlike the in operator, this method does not check down the object's prototype chain

看描述,就是我们想要的东西。

在for...in..里做个 判断就OK了

if(a.hasOwnProperty(p)){  

            document.write(p+"="+a[p]+"<br/>");  

        }

另外,附上hasOwnProperty用法示例,来源于互联网:

function Book(title, author) {  

   this.title = title;  

   this.author = author;  

  }  

   Book.prototype.price = 9.99;  

   Object.prototype.copyright = "herongyang.com";  

   var myBook = new Book("JavaScript Tutorials", "Herong Yang");  

   // Dumping built-in properties at the base prototype level  

   document.writeln("/nObject.prototype's built-in properties:");  

   dumpProperty(Object.prototype, "constructor");  

   dumpProperty(Object.prototype, "hasOwnProperty");  

   dumpProperty(Object.prototype, "isPrototypeOf");  

   dumpProperty(Object.prototype, "toString");  

   dumpProperty(Object.prototype, "valueOf");  

   dumpProperty(Object.prototype, "copyright");  

   // Dumping built-in properties at the my prototype level  

   document.writeln("/n==================/nBook.prototype's built-in properties:");  

   dumpProperty(Book.prototype, "constructor");  

   dumpProperty(Book.prototype, "hasOwnProperty");  

   dumpProperty(Book.prototype, "isPrototypeOf");  

   dumpProperty(Book.prototype, "toString");  

   dumpProperty(Book.prototype, "valueOf");  

   dumpProperty(Book.prototype, "copyright");  

   // Dumping built-in properties at the object level  

   document.writeln("/n==================/nmyBook's built-in properties:");  

   dumpProperty(myBook, "constructor");  

   dumpProperty(myBook, "hasOwnProperty");  

   dumpProperty(myBook, "isPrototypeOf");  

   dumpProperty(myBook, "toString");  

   dumpProperty(myBook, "valueOf");  

   dumpProperty(myBook, "copyright");  

function dumpProperty(object, property) {  

   var inheritance;   

   if (object.hasOwnProperty(property))   

      inheritance = "Local";  

   else  

      inheritance = "Inherited";  

   document.writeln(property+": "+inheritance+": " 

      +object[property]);  

}

查看浏览器支持javascript到哪个版本:

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">  

<html xmlns="http://www.w3.org/1999/xhtml">  

<head>  

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />  

  <title>浏览器的JavaScript版本支持测试</title>  

 </head>  

 <body>  

   <script language="JavaScript">    

   //document.write("您的浏览器类型:"+navigator.appName+"<br/>");  

   //document.write("浏览器版本:"+navigator.appVersion+"<br/>");  

      //支持JavaScript1.0的浏览器才能够执行该脚本    

      document.write('该浏览器支持JavaScript1.0<br/>');    

  </script>    

  <script language="JavaScript1.1">    

      //支持JavaScript1.1的浏览器才能够执行该脚本   

      document.write('该浏览器支持JavaScript1.1<br/>');    

  </script>    

  <script language="JavaScript1.2">    

      //支持JavaScript1.2的浏览器才能够执行该脚本    

      document.write('该浏览器支持JavaScript1.2<br/>');    

  </script>    

  <script language="JavaScript1.3">    

      //支持JavaScript1.3的浏览器才能够执行该脚本   

      document.write('该浏览器支持JavaScript1.3<br/>');    

  </script>    

  <script language="JavaScript1.4">    

      //支持JavaScript1.4的浏览器才能够执行该脚本    

      document.write('该浏览器支持JavaScript1.4<br/>');    

  </script>    

  <script language="JavaScript1.5">    

      //支持JavaScript1.5的浏览器才能够执行该脚本    

      document.write('该浏览器支持JavaScript1.5<br/>');    

  </script>  

  <script language="JavaScript1.6">    

      //支持JavaScript1.6的浏览器才能够执行该脚本    

      document.write('该浏览器支持JavaScript1.6<br/>');    

  </script>  

  <script   language="JavaScript1.7">    

      //支持JavaScript1.7的浏览器才能够执行该脚本    

      document.write('该浏览器支持JavaScript1.7<br/>');    

  </script>  

  <script language="JavaScript1.8">    

      //支持JavaScript 1.8的浏览器才能够执行该脚本    

      document.write('该浏览器支持JavaScript1.8<br/>');    

  </script>  

  <script language="JavaScript1.9">    

      //支持JavaScript1.9的浏览器才能够执行该脚本    

      document.write('该浏览器支持JavaScript1.9<br/>');    

  </script>   

 </body>  

</html>
Javascript 相关文章推荐
jquery 获取json数据实现代码
Apr 27 Javascript
jquery 获取 outerHtml 包含当前节点本身的代码
Oct 30 Javascript
JavaScript中具名函数的多种调用方式总结
Nov 08 Javascript
js判断浏览器版本以及浏览器内核的方法
Jan 20 Javascript
使用Node.js实现HTTP 206内容分片的教程
Jun 23 Javascript
Javascript实现字数统计
Jul 03 Javascript
javascript中判断json的方法总结
Aug 27 Javascript
实例讲解jquery与json的结合
Jan 07 Javascript
详解如何在nuxt中添加proxyTable代理
Aug 10 Javascript
layui 实现表格某一列显示图标
Sep 19 Javascript
支付宝小程序实现省市区三级联动
Jun 21 Javascript
javascript利用canvas实现鼠标拖拽功能
Jul 23 Javascript
JavaScript的内存释放问题详解
Jan 21 #Javascript
script标签属性用type还是language
Jan 21 #Javascript
JS交换变量的方法
Jan 21 #Javascript
setinterval()与clearInterval()JS函数的调用方法
Jan 21 #Javascript
js实现DOM走马灯特效的方法
Jan 21 #Javascript
浅谈javascript 迭代方法
Jan 21 #Javascript
js实现用户注册协议倒计时的方法
Jan 21 #Javascript
You might like
php面向对象全攻略 (二) 实例化对象 使用对象成员
2009/09/30 PHP
PHP防范SQL注入的具体方法详解(测试通过)
2014/05/09 PHP
php获取根域名方法汇总
2014/10/28 PHP
关于php 高并发解决的一点思路
2017/04/16 PHP
Laravel框架分页实现方法分析
2018/06/12 PHP
PHP实现小程序批量通知推送
2018/11/27 PHP
浅谈PHP中的那些魔术常量
2020/12/02 PHP
在IE,Firefox,Safari,Chrome,Opera浏览器上调试javascript
2008/12/02 Javascript
javascript整除实现代码
2010/11/23 Javascript
javascript不可用的问题探究
2013/10/01 Javascript
jQuery学习笔记之 Ajax操作篇(三) - 过程处理
2014/06/23 Javascript
JavaScript严格模式禁用With语句的原因
2014/10/20 Javascript
JavaScript forEach()遍历函数使用及介绍
2015/07/08 Javascript
JS实现黑色大气的二级导航菜单效果
2015/09/18 Javascript
原生javascript实现的ajax异步封装功能示例
2016/11/03 Javascript
使用vue.js实现联动效果的示例代码
2017/01/10 Javascript
原生js实现弹出层效果
2017/01/20 Javascript
折叠菜单及选择器的运用
2017/02/03 Javascript
利用策略模式与装饰模式扩展JavaScript表单验证功能
2017/02/14 Javascript
js模块加载方式浅析
2017/08/12 Javascript
jQuery实现弹窗下底部页面禁止滑动效果
2017/12/19 jQuery
解决vue2中使用elementUi打包报错的问题
2020/09/22 Javascript
关于Vue中$refs的探索浅析
2020/11/05 Javascript
elementUI同一页面展示多个Dialog的实现
2020/11/19 Javascript
Python用zip函数同时遍历多个迭代器示例详解
2016/11/14 Python
python递归删除指定目录及其所有内容的方法
2017/01/13 Python
Python使用Django实现博客系统完整版
2020/09/29 Python
python微信聊天机器人改进版(定时或触发抓取天气预报、励志语录等,向好友推送)
2019/04/25 Python
django中瀑布流写法实例代码
2019/10/14 Python
python实现图像拼接功能
2020/03/23 Python
python调用摄像头的示例代码
2020/09/28 Python
如何实现jdbc性能优化
2012/07/30 面试题
党员批评与自我批评发言稿
2014/10/14 职场文书
先进个人评语大全
2015/01/04 职场文书
堂吉诃德读书笔记
2015/06/30 职场文书
Spring Security中用JWT退出登录时遇到的坑
2021/10/16 Java/Android