1.使用Object的直接实例构造轻量级的字典

字典就是可变长的字符串与值得映射集合。JavaScript甚至提供了枚举一个对象属性名的利器——for...in循环。

var dict = { alice : 34 , bob : 24 , chris : 62 } ;
var people = [ ] ;
for(var name in dict){
people.push(name + ":"+ dict [name])
}
people ;

坚持Object的直接实例原则可以使得for...in循环摆脱原型污染的影响。

2.防止原型污染的最简单的方式之一就是一开始就不适用原型。但在ES5之前没有标准的方式创建一个空原型的新对象。你可能会尝试设置一个构造函数的原型属性为null或者undefined。

function C() { }
C.prototype = null ;

但实例化该构造函数仍然得到的是Object的实例。

var o = new C() ;
Object.getPrototypeOf(o) === null ;
Object.getPrototypeOf(o) === Object.prototype ;

ES5首先提供了标准方法来创建一个没有原型的对象。Object.create函数能够使用一个用户指定的原型链和一个属性描述符动态地构造对象。通过简单的传递一个null原型参数和一个空的描述符,我们就可以建立一个真正的空对象。

var x = Object.create(null) ;
Object.getPrototypeOf(o) === null ;

原型污染无法影响这样的对象行为。一些不支持Object.create函数的旧的JavaScript环境可能支持另一种值得一提的方式。在许多环境中,__proto__提供了对对象内部原型链的读写访问。对象字面量语法也支持初始化一个原型链为null的新对象。

3.使用hasOwnProperty方法以避免原型污染

JavaScript的对象操作总是以继承的方式工作。即使是一个空的对象字面量也继承了Object.prototype的大量属性。但它提供的hasOwnProperty可以避免原型污染。

4.绝对不要在Object.prototype中增加可枚举的属性

到目前为止,for...in循环最常见的用法是枚举字典中的元素。如果想允许对字典对象使用for...in循环,那么不要在共享的Object.prototype中增加可枚举的属性。

如下代码:

Object.prototype.allKeys = function(){
var result = [ ] ;
for(var key in this){
result.push(key) ;
}
return result ;
}

遗憾的是,该方法污染了自身。如果你真的想在Object.prototype中增加属性,ES5提供了一种更友好的机制。Object.defineProperty方法可以定义一个对象的属性并指定该属性的元数据。例如,我们可以用与之前完全一样的方式定义上面的属性而通过设置可枚举属性为false使其在for...in循环中不可见。

Object.defineProperty(Object.prototype , “allkeys”,{
value : function(){
var result = [ ] ;
for (var key in this){
result.push(key) ;
}
return result ;
},
writable : true ,
enumerable : false ,
configurable : true
});

每当你增加一个不应该在for...in循环中出现的属性时,Object.defineProperty便是你的选择。

 

5.避免在枚举期间修改对象

事实上,ECMAScript标准规定了:

如果被枚举的对象在枚举期间添加了新的属性,那么在枚举期间并不能保证新添加的属性能够被访问。当迭代一个对象时,如果该对象的内容可能会在循环期间被改变,应该使用while循环或经典的for循环来代替for...in循环。

 

6.数组迭代要优先使用for循环而不是for...in循环。

7.迭代方法优于循环。

①JavaScript的闭包是一种为这些模式建立迭代抽象方便的、富有表现力的手法,从而使我们避免复制,粘贴循环头部。ES5为最常用的一些模式提供了便利的方法。Array.prototype.forEach是其中最简单的一个。例如:

for(var i = 0 , n = players.length ; i < n ; i ++){
players[ i ].score++ ;
}

我们可以用下面的代码替换上面的循环:

players.forEach(function(p){
p.score++ ;
}) ;

这段代码不仅更简单可读,而且还消除了终止条件和任何数组索引。

②另一种常见的模式是对数组的每个元素进行一些操作后建立一个新的数组。ES5引入了Array.prototype.map方法使该模式更简单、更优雅。

var trimmed = input.map(function(s){
return s.trim() ;
}) ;

③另一种常见的模式是计算一个新的数组,该数组只包含现有数组的一些元素。Array.prototype.filter使其变得很简便。它需要一个谓词,该谓词是一个函数,如果元素应该存在于新数组中则返回真值否则返回假值。例如,我们可以从价格表中提取出符合一个特定价格区间的列表:

listings.filter(function(listing){
return listing.price >= min && listing.price <=max ;
}) ;

循环只有一点优于迭代函数,那就是前者有控制流操作,如break和continue。

此外,ES5的数组方法some和every可以用于提前终止循环。

8.在类数组对象上复用通用的数组方法

Array.ptototype中的标准方法被设计成其他对象可复用的方法,即使这些对象并没有继承Array。事实证明,许多这样的类数组对象接踵而至地出现在JavaScript的不同地方。在Web平台,DOM的NodeList类是另一个类数组对象的实例。类似document.getElementsByTagName这样的操作会查询Web页面中的节点,并返回NodeList作为搜索的结果。

Effective JavaScript :第五章的更多相关文章

  1. [Effective Java]第五章 泛型

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  2. Effective JavaScript :第二章

    1.熟练掌握闭包 理解闭包要学会三个基本的事实: ①JavaScript允许你引用在当前函数以外定义的变量: 例如: function makeSandwich(){ var magicIngredi ...

  3. Effective JavaScript :第一章

    第一章 一.严格模式与非严格模式 1.在程序中启用严格模式的方式是在程序的最开始增加一个特定的字符串字面量: ‘use strict’ 同样可以在函数体的开始处加入这句指令以启用该函数的严格模式. f ...

  4. Javascript 第五章总结:A trip to Objectville

    前言 在以前的代码中,我们使用 primitive 类型的变量和 procedural manner 来执行脚本.但是,更好的办法是 object-oriented (面向对象)的.作者说:它能让我们 ...

  5. javascript第五章--函数表达式

    ① 递归 ② 闭包 ③ 模仿块级作用域 ④ 私有变量

  6. 读《编写可维护的JavaScript》第五章总结

    第五章 UI层的松耦合 5.1 什么是松耦合 在Web开发中,用户界面是由三个彼此隔离又相互作用的层定义的: HTML是用来定义页面的数据和语义 CSS用来给页面添加样式 JavaScript用来给页 ...

  7. Javascript权威指南——第二章词法结构,第三章类型、值和变量,第四章表达式和运算符,第五章语句

    第二章 词法结构 一.HTML并不区分大小写(尽管XHTML区分大小写),而javascript区分大小写:在HTML中,这些标签和属性名可以使用大写也可以使用小写,而在javascript中必须小写 ...

  8. [Effective JavaScript 笔记]第3章:使用函数--个人总结

    前言 这一章把平时会用到,但不会深究的知识点,分开细化地讲解了.里面很多内容在高3等基础内容里,也有很多讲到.但由于本身书籍的篇幅较大,很容易忽视对应的小知识点.这章里的许多小提示都很有帮助,特别是在 ...

  9. 【JavaScript权威指南(第五版)】笔记之第二部分 客户端JavaScript 第13章~第23章

    第十三章 Web浏览器中的javascript ①   eg:下面两行代码实际上执行的是相同的功能 var answer = 42; window.answer = 42;   ③每个window对象 ...

  10. 【JavaScript权威指南(第五版)】笔记之第一部分 核心javascript (第1章~第12章)

    第一章 javascript概述 ①.javascript是一种松散类型语言;也是一种解释型语言;         第二章 词法结构 ①.大小写敏感         第三章 数据类型和值 ①.isFi ...

随机推荐

  1. vue.js之个人总结

    1.MVVM模式 MVVM模式(Model-View-ViewModel)的运作如下图: 1)上图解析:ViewModel是Vue.js的核心,它是一个Vue实例.Vue实例是作用于某一个HTML元素 ...

  2. 关于map()与filter()

    def f1(a): if a == 3: pass else: return a*a li = [1,2,3,4] for i in filret(f1,li) print(i) def f1(a) ...

  3. leetcode 024

    Given a linked list, swap every two adjacent nodes and return its head. For example,Given 1->2-&g ...

  4. Reprint: Serialization

    Having just recently ran into some major serialization issues I’m going to list some of the errors a ...

  5. iOS二维码生成与识别

    在 IOS7 以前,在IOS中实现二维码和条形码扫描,有两大开源组件 ZBar 与 ZXing. 总结下各自的缺点: ZBar在扫描的灵敏度上,和内存的使用上相对于ZXing上都是较优的,但是对于 & ...

  6. img 转化成iso镜像的办法

    最近在使用KVM启用虚拟机,然后将里面的环境和配置 配置成我们公司需要的环境,再打包成iso镜像,之后再次生成新的虚拟机. 但是KVM启动出的镜像生成的是img镜像 ,需要将img镜像转换成iso镜像 ...

  7. re模块 | Python 3.5

    https://docs.python.org/3/library/re.html http://www.cnblogs.com/PythonHome/archive/2011/11/19/22554 ...

  8. hdu 4122 Alice's mooncake shop(单调队列)

    题目链接:hdu 4122 Alice's mooncake shop 题意: 有n个订单和可以在m小时内制作月饼 接下来是n个订单的信息:需要在mon月,d日,year年,h小时交付订单r个月饼 接 ...

  9. 关于pagerank算法的一点点总结

    1. PageRank算法每个顶点收敛的值与每个点的初值是没有关系的,每个点随便赋初值. 2.像q=0.8这样的阻尼系数已经解决了PageRank中处在的孤立点问题.黑洞效应问题. 3.当有那个点进行 ...

  10. JS正则表达式的test()方法检查汉字

    汉字的检查: var reg1=/[^\u4e00-\u9fa5]/g; var text="正则表达式1"; var res1=reg1.test(text);//为true表示 ...