js对象的核心是一个字符串属性名与属性值的映射表。使用对象实现字典易如反掌,字典是可变长的字符串与值的映射集合。

for...in

js提供了枚举一个对象属性名的利器--for...in循环。

var dict={zhangsan:34,lisi:24,wangwu:62};
var people=[];
for(var name in dict){
people.push(name+":"+dict[name]);
}
people;//["zhangsan:34", "lisi:24", "wangwu:62"]

使用自定义对象

但一些自定义的对象还会继承其原型对象中的属性,for...in循环除了枚举"自身"属性外,原型链中的属性也不会放过。下面看一下例子

function NaiveDict(){
}
NaiveDict.prototype.count=function(){
var i=0;
for(var name in this){
i++;
}
return i;
};
NaiveDict.prototype.toString=function(){
return '[Object NaiveDict]';
};
var dict=new NaiveDict();
dict.zhangsan=24;
dict.lisi=34;
dict.wangwu=62;
dict.count();//5

上面的代码,实例化一个dict对象,并添加了私有属性(zhangsan,lisi,wangwu)并继承了原型对象的方法(toString,count),所以当调用count方法时,这个时候的for...in循环,不仅枚举了了私有属性也枚举了原型对象的方法属性(zhangsan,lisi,wangwu,toString,count)。

使用数组类型

利用js的灵活性,我们可以给任意的类型的对象添加属性,因此给数组添加属性似乎能工作。

var dict=new Array();
dict.zhangsan=24;
dict.lisi=34;
dict.wangwu=62;
dict.zhangsang;//24

当我们给数组的原型对象添加一些原型方法,也就是前面讲的猴子补丁ployfill的使用时,这个时候我们再枚举上面的dict对象的属性时,错误就出现了。

Array.prototype.first=function(){
return this[0];
};
Array.prototype.last=function(){
return this[this.length-1];
};
Array.prototype.eq=function(idx){
if(idx<0)idx=this.length+idx;
return this[idx];
};

运行一下枚举

var names=[];
for(var name in dict){
names.push(name);
}
names;//["zhangsan", "lisi", "wangwu","first","last"]

轻量级字典的正确姿势

首要原则

应该仅仅将Object的直接实例作为字典,而不是子类,或其它对象。
上面的数组的例子可以改成下面这些的代码

var dict={};
dict.zhangsan=24;
dict.lisi=34;
dict.wangwu=62;
dict.zhangsang;
var names=[];
for(var name in dict){
names.push(name);
}
names;//["zhangsan", "lisi", "wangwu"]

需要注意

还是无法避免对于Object.prototype对象的修改,但可以将风险仅仅局限在Object.prototype。

优点

虽说所有人都可以修改Object.prototype对象,也会对for...in循环造成影响。但相比之下,增加属性到Array.prototype中是合理的。如之前给不支持数组标准方法的环境中将这些方法增加到Array.prototype中。这些属性也会导致for...in循环,坚持Object的直接实例原则,可以使得for...in循环摆脱原型污染的影响。对于构建行为正确的字典,这是个必要非充分条件。

提示

  • 使用对象字面量构建轻量级字典

  • 轻量级字典应该是Object.prototype的直接子类,以使for...in循环免受原型的污染

附录:for...in

以下内容来自:Mozilla 开发者社区
以任意序迭代一个对象的可枚举属性。每个不同的属性,语句都会被执行一次。

语法

for(variable in object){...}

参数

  • variable:每次迭代,一个不同的属性名将赋予variable

  • object:可枚举属性被迭代的对象

描述

  • 只言遍历可枚举属性。

  • 循环将迭代对象的所有可枚举属性和从它的构造函数的prototype继承而来的

删除、添加或修改属性

for...in循环以任意序迭代一个对象的属性。如果一个属性在一次迭代中被修改,在稍后被访问,其在循环中的值是其在稍后时间的值。一个在被访问之前已经被删除的属性将不会在之后被访问。在迭代进行时被添加到对象的属性,可能在之后的迭代被访问,也可能被忽略。通常,在迭代过程中最好不要在对象上进行添加、修改或删除属性的操作,除非是对当前正在被访问的属性。因为访问的无序性,都无法保证,添加、修改或删除的属性,会在之后被访问。

Array迭代和for...in

数组索引是可枚举的整数名,其他方面和普通对象属性没有区别。for...in的无序性,无法保证枚举是按索引顺序进行,但会返回所有可枚举的属性,包括非整数名称和继承的属性。
如果次序很重要,可以使用for循环来迭代数组。

仅迭代自身的属性

只需要对象本身的属性,不包括它继承来的。可以使用getOwnPropertyNames()或执行hasOwnProperty()来确定某属性是否是对象自身的。

例子

返回一个对象的所有可枚举属性
var obj={a:2,b:1,c:30};
for(var prop in obj){
console.log('obj.'+prop+'='+obj[prop]);
}
// "obj.a = 2"// "obj.b = 1"// "obj.c = 30"
只返回对象自身的可枚举属性
var triangle = {a:1, b:2, c:3};

function ColoredTriangle() {
this.color = "red";
} ColoredTriangle.prototype = triangle; var obj = new ColoredTriangle(); for (var prop in obj) {
if( obj.hasOwnProperty( prop ) ) {
console.log("o." + prop + " = " + obj[prop]);
}
}
// "o.color = red"

[Effective JavaScript 笔记]第43条:使用Object的直接实例构造轻量级的字典的更多相关文章

  1. [Effective JavaScript 笔记] 第4条:原始类型优于封闭对象

    js有5种原始值类型:布尔值.数字.字符串.null和undefined. 用typeof检测一下: typeof true; //"boolean" typeof 2; //&q ...

  2. [Effective JavaScript 笔记]第28条:不要信赖函数对象的toString方法

    js函数有一个非凡的特性,即将其源代码重现为字符串的能力. (function(x){ return x+1 }).toString();//"function (x){ return x+ ...

  3. [Effective JavaScript 笔记] 第5条:避免对混合类型使用==运算符

    “1.0e0”=={valueOf:function(){return true;}} 是值是多少? 这两个完全不同的值使用==运算符是相等的.为什么呢?请看<[Effective JavaSc ...

  4. [Effective JavaScript 笔记]第27条:使用闭包而不是字符串来封装代码

    函数是一种将代码作为数据结构存储的便利方式,代码之后可以被执行.这使得富有表现力的高阶函数抽象如map和forEach成为可能.它也是js异步I/O方法的核心.与此同时,也可以将代码表示为字符串的形式 ...

  5. [Effective JavaScript 笔记]第45条:使用hasOwnProperty方法以避免原型污染

    之前的43条,44条讨论了属性的枚举,但都没有彻底地解决属性查找中原型污染的问题.看下面关于字典的一些操作 'zhangsan' in dict; dict.zhangsan; dict.zhangs ...

  6. [Effective JavaScript 笔记]第44条:使用null原型以防止原型污染

    第43条中讲到的就算是用了Object的直接实例,也无法完全避免,Object.prototype对象修改,造成的原型污染.防止原型污染最简单的方式之一就是不使用原型.在ES5之前,并没有标准的方式创 ...

  7. [Effective JavaScript 笔记]第38条:在子类的构造函数中调用父类的构造函数

    示例 场景类 场景图(scene)是在可视化的过程中(如游戏或图形仿真场景)描述一个场景的对象集合.一个简单的场景包含了在该场景中的所有对象(称角色),以及所有角色的预加载图像数据集,还包含一个底层图 ...

  8. [Effective JavaScript 笔记]第47条:绝不要在Object.prototype中增加可枚举的属性

    之前的几条都不断地重复着for...in循环,它便利好用,但又容易被原型污染.for...in循环最常见的用法是枚举字典中的元素.这里就是从侧面提出不要在共享的Object.prototype中增加可 ...

  9. [Effective JavaScript 笔记]第31条:使用Object.getPrototypeOf函数而不要使用__proto__属性

    ES5引入Object.getPrototypeOf函数作为获取对象原型的标准API,但由于之前的很多js引擎使用了一个特殊的__proto__属性来达到相同的目的.但有些浏览器并不支持这个__pro ...

随机推荐

  1. WCF Data Service 使用小结 (一)—— 了解OData协议

    最近做了一个小项目,其中用到了 WCF Data Service,之前是叫 ADO.NET Data Service 的.关于WCF Data Service,博客园里的介绍并不多,但它确实是个很好的 ...

  2. excel导入数据库

    日常工作中,感觉一些基础知识需要做下笔记,可能是刚毕业的缘故吧,还保持着做笔记的习惯,但根据以往经验,纸质笔记最多保持一年,过后想找已是难过登天.电子版笔记感觉很不错,尤其是发布到网络中.笔记内容是本 ...

  3. web前端开发教程系列-1 - 前端开发编辑器介绍

    目录: 前言 一. Webstorm 1. 优点 2. 缺点 3. 教程 4. 插件 5. 技巧 二. SublimeText 1. 优点 2. 缺点 3. 教程 4. 插件 5. 技巧 前言 由于很 ...

  4. “耐撕”团队 2016.04.11 站立会议

    1. 时间 : 19:30--19:50. 共计20分钟. 2. 成员 : Z 郑蕊 * 组长 (博客:http://www.cnblogs.com/zhengrui0452/), P 濮成林(博客: ...

  5. iOS开发小技巧--高斯模糊框架的应用

    事件背景:彩票项目中点击检查更新之后的操作,高斯模糊效果并弹出HUD 注意:在应用别人的框架的时候,最好封装一下下. 新建一个类  继承自高斯模糊的类. 使用方法:新建一个高斯模糊类的View,添加到 ...

  6. Svn-在eclipse中安装svn插件

    在eclipse中安装svn有两种方式 1:直接下载svn的插件包安装 使用的版本为1.8.x Links for 1.8.x Release: Eclipse update site URL: ht ...

  7. Spring-事物配置

    Spring框架支持事务管理的核心是事务管理器抽象,对于不同的数据访问框架(如Hibernate)通过实现策略接口PlatformTransactionManager,从而能支持各种数据访问框架的事务 ...

  8. VB中的属性、方法和事件概念解析

    Visual Basic 语言中的所有对象都有它们自己的属性.方法和事件,其中包括窗体和控件.可以将属性视为对象的特性,将方法视为对象的操作,而将事件视为对象的响应. 日常生活中的对象(如氦气球)也具 ...

  9. 【FE前端学习】第二阶段任务-基础

    技能学习部分: 1.需要熟练掌握HTML标签以及CSS各个常用属性. 2.掌握CSS3 常用属性 3.掌握jquery的基本用法,对于JS基本逻辑语句需要熟练掌握 上文 [FE前端学习]第二阶段任务- ...

  10. com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException: Unknown database 'db_item'

    一直很奇怪,为什么报错,进入mysql命令行,show databases:发现多谢了一个空格,如上图.