深入理解JavaScript系列(6) 强大的原型和原型链


Posted in Javascript onJanuary 15, 2012

前言
JavaScript 不包含传统的类继承模型,而是使用 prototypal 原型模型。

虽然这经常被当作是 JavaScript 的缺点被提及,其实基于原型的继承模型比传统的类继承还要强大。实现传统的类继承模型是很简单,但是实现 JavaScript 中的原型继承则要困难的多。

由于 JavaScript 是唯一一个被广泛使用的基于原型继承的语言,所以理解两种继承模式的差异是需要一定时间的,今天我们就来了解一下原型和原型链。

原型
10年前,我刚学习JavaScript的时候,一般都是用如下方式来写代码:

var decimalDigits = 2, 
tax = 5; function add(x, y) { 
return x + y; 
} 
function subtract(x, y) { 
return x - y; 
} 
//alert(add(1, 3));

通过执行各个function来得到结果,学习了原型之后,我们可以使用如下方式来美化一下代码。

原型使用方式1:
在使用原型之前,我们需要先将代码做一下小修改:

var Calculator = function (decimalDigits, tax) { 
this.decimalDigits = decimalDigits; 
this.tax = tax; 
};

然后,通过给Calculator对象的prototype属性赋值对象字面量来设定Calculator对象的原型。
Calculator.prototype = { 
add: function (x, y) { 
return x + y; 
}, subtract: function (x, y) { 
return x - y; 
} 
}; 
//alert((new Calculator()).add(1, 3));

这样,我们就可以new Calculator对象以后,就可以调用add方法来计算结果了。

原型使用方式2:
第二种方式是,在赋值原型prototype的时候使用function立即执行的表达式来赋值,即如下格式:

Calculator.prototype = function () { } ();
它的好处在前面的帖子里已经知道了,就是可以封装私有的function,通过return的形式暴露出简单的使用名称,以达到public/private的效果,修改后的代码如下:

Calculator.prototype = function () { 
add = function (x, y) { 
return x + y; 
}, subtract = function (x, y) { 
return x - y; 
} 
return { 
add: add, 
subtract: subtract 
} 
} (); 
//alert((new Calculator()).add(11, 3));

同样的方式,我们可以new Calculator对象以后调用add方法来计算结果了。

再来一点
分步声明:
上述使用原型的时候,有一个限制就是一次性设置了原型对象,我们再来说一下如何分来设置原型的每个属性吧。

var BaseCalculator = function () { 
//为每个实例都声明一个小数位数 
this.decimalDigits = 2; 
}; //使用原型给BaseCalculator扩展2个对象方法 
BaseCalculator.prototype.add = function (x, y) { 
return x + y; 
}; 
BaseCalculator.prototype.subtract = function (x, y) { 
return x - y; 
};

首先,声明了一个BaseCalculator对象,构造函数里会初始化一个小数位数的属性decimalDigits,然后通过原型属性设置2个function,分别是add(x,y)和subtract(x,y),当然你也可以使用前面提到的2种方式的任何一种,我们的主要目的是看如何将BaseCalculator对象设置到真正的Calculator的原型上。
var BaseCalculator = function() { 
this.decimalDigits = 2; 
}; BaseCalculator.prototype = { 
add: function(x, y) { 
return x + y; 
}, 
subtract: function(x, y) { 
return x - y; 
} 
};

创建完上述代码以后,我们来开始:
var Calculator = function () { 
//为每个实例都声明一个税收数字 
this.tax = 5; 
}; Calculator.prototype = new BaseCalculator();

我们可以看到Calculator的原型是指向到BaseCalculator的一个实例上,目的是让Calculator集成它的add(x,y)和subtract(x,y)这2个function,还有一点要说的是,由于它的原型是BaseCalculator的一个实例,所以不管你创建多少个Calculator对象实例,他们的原型指向的都是同一个实例。
var calc = new Calculator(); 
alert(calc.add(1, 1)); 
//BaseCalculator 里声明的decimalDigits属性,在 Calculator里是可以访问到的 
alert(calc.decimalDigits);

上面的代码,运行以后,我们可以看到因为Calculator的原型是指向BaseCalculator的实例上的,所以可以访问他的decimalDigits属性值,那如果我不想让Calculator访问BaseCalculator的构造函数里声明的属性值,那怎么办呢?这么办:
var Calculator = function () { 
this.tax= 5; 
}; Calculator.prototype = BaseCalculator.prototype;

通过将BaseCalculator的原型赋给Calculator的原型,这样你在Calculator的实例上就访问不到那个decimalDigits值了,如果你访问如下代码,那将会提升出错。
var calc = new Calculator(); 
alert(calc.add(1, 1)); 
alert(calc.decimalDigits);

重写原型:
在使用第三方JS类库的时候,往往有时候他们定义的原型方法是不能满足我们的需要,但是又离不开这个类库,所以这时候我们就需要重写他们的原型中的一个或者多个属性或function,我们可以通过继续声明的同样的add代码的形式来达到覆盖重写前面的add功能,代码如下:
//覆盖前面Calculator的add() function 
Calculator.prototype.add = function (x, y) { 
return x + y + this.tax; 
}; var calc = new Calculator(); 
alert(calc.add(1, 1));

这样,我们计算得出的结果就比原来多出了一个tax的值,但是有一点需要注意:那就是重写的代码需要放在最后,这样才能覆盖前面的代码。

原型链
在将原型链之前,我们先上一段代码:

function Foo() { 
this.value = 42; 
} 
Foo.prototype = { 
method: function() {} 
}; function Bar() {} 
// 设置Bar的prototype属性为Foo的实例对象 
Bar.prototype = new Foo(); 
Bar.prototype.foo = 'Hello World'; 
// 修正Bar.prototype.constructor为Bar本身 
Bar.prototype.constructor = Bar; 
var test = new Bar() // 创建Bar的一个新实例 
// 原型链 
test [Bar的实例] 
Bar.prototype [Foo的实例] 
{ foo: 'Hello World' } 
Foo.prototype 
{method: ...}; 
Object.prototype 
{toString: ... /* etc. */};

上面的例子中,test 对象从 Bar.prototype 和 Foo.prototype 继承下来;因此,它能访问 Foo 的原型方法 method。同时,它也能够访问那个定义在原型上的 Foo 实例属性 value。需要注意的是 new Bar() 不会创造出一个新的 Foo 实例,而是重复使用它原型上的那个实例;因此,所有的 Bar 实例都会共享相同的 value 属性。

属性查找:
当查找一个对象的属性时,JavaScript 会向上遍历原型链,直到找到给定名称的属性为止,到查找到达原型链的顶部 - 也就是 Object.prototype - 但是仍然没有找到指定的属性,就会返回 undefined,我们来看一个例子:

function foo() { 
this.add = function (x, y) { 
return x + y; 
} 
} foo.prototype.add = function (x, y) { 
return x + y + 10; 
} 
Object.prototype.subtract = function (x, y) { 
return x - y; 
} 
var f = new foo(); 
alert(f.add(1, 2)); //结果是3,而不是13 
alert(f.subtract(1, 2)); //结果是-1

通过代码运行,我们发现subtract是安装我们所说的向上查找来得到结果的,但是add方式有点小不同,这也是我想强调的,就是属性在查找的时候是先查找自身的属性,如果没有再查找原型,再没有,再往上走,一直插到Object的原型上,所以在某种层面上说,用 for in语句遍历属性的时候,效率也是个问题。

还有一点我们需要注意的是,我们可以赋值任何类型的对象到原型上,但是不能赋值原子类型的值,比如如下代码是无效的:

function Foo() {} 
Foo.prototype = 1; // 无效

hasOwnProperty函数:
hasOwnProperty是Object.prototype的一个方法,它可是个好东西,他能判断一个对象是否包含自定义属性而不是原型链上的属性,因为hasOwnProperty 是 JavaScript 中唯一一个处理属性但是不查找原型链的函数。
// 修改Object.prototype 
Object.prototype.bar = 1; 
var foo = {goo: undefined}; foo.bar; // 1 
'bar' in foo; // true 
foo.hasOwnProperty('bar'); // false 
foo.hasOwnProperty('goo'); // true

只有 hasOwnProperty 可以给出正确和期望的结果,这在遍历对象的属性时会很有用。 没有其它方法可以用来排除原型链上的属性,而不是定义在对象自身上的属性。

但有个恶心的地方是:JavaScript 不会保护 hasOwnProperty 被非法占用,因此如果一个对象碰巧存在这个属性,就需要使用外部的 hasOwnProperty 函数来获取正确的结果。

var foo = { 
hasOwnProperty: function() { 
return false; 
}, 
bar: 'Here be dragons' 
}; foo.hasOwnProperty('bar'); // 总是返回 false 
// 使用{}对象的 hasOwnProperty,并将其上下为设置为foo 
{}.hasOwnProperty.call(foo, 'bar'); // true

当检查对象上某个属性是否存在时,hasOwnProperty 是唯一可用的方法。同时在使用 for in loop 遍历对象时,推荐总是使用 hasOwnProperty 方法,这将会避免原型对象扩展带来的干扰,我们来看一下例子:
// 修改 Object.prototype 
Object.prototype.bar = 1; var foo = {moo: 2}; 
for(var i in foo) { 
console.log(i); // 输出两个属性:bar 和 moo 
}

我们没办法改变for in语句的行为,所以想过滤结果就只能使用hasOwnProperty 方法,代码如下:
// foo 变量是上例中的 
for(var i in foo) { 
if (foo.hasOwnProperty(i)) { 
console.log(i); 
} 
}

这个版本的代码是唯一正确的写法。由于我们使用了 hasOwnProperty,所以这次只输出 moo。如果不使用 hasOwnProperty,则这段代码在原生对象原型(比如 Object.prototype)被扩展时可能会出错。

总结:推荐使用 hasOwnProperty,不要对代码运行的环境做任何假设,不要假设原生对象是否已经被扩展了。

总结
原型极大地丰富了我们的开发代码,但是在平时使用的过程中一定要注意上述提到的一些注意事项。

Javascript 相关文章推荐
Prototype使用指南之dom.js
Jan 10 Javascript
addEventListener和attachEvent二者绑定的执行函数中的this不相同
Dec 09 Javascript
node.js Web应用框架Express入门指南
May 28 Javascript
javascript原生和jquery库实现iframe自适应高度和宽度
Jul 18 Javascript
浅析JavaScript事件和方法
Feb 28 Javascript
JavaScript使用cookie实现记住账号密码功能
Apr 27 Javascript
JavaScript SHA512&SHA256加密算法详解
Aug 11 Javascript
原生JS实现首页进度加载动画
Sep 14 Javascript
利用纯JS实现像素逐渐显示的方法示例
Aug 14 Javascript
python爬取安居客二手房网站数据(实例讲解)
Oct 19 Javascript
浅谈Node 异步IO和事件循环
May 05 Javascript
Vue的props父传子的示例代码
May 20 Javascript
深入理解JavaScript系列(4) 立即调用的函数表达式
Jan 15 #Javascript
深入理解JavaScript系列(3) 全面解析Module模式
Jan 15 #Javascript
深入理解JavaScript系列(2) 揭秘命名函数表达式
Jan 15 #Javascript
深入理解JavaScript系列(1) 编写高质量JavaScript代码的基本要点
Jan 15 #Javascript
Prototype源码浅析 String部分(三)之HTML字符串处理
Jan 15 #Javascript
Prototype源码浅析 String部分(一)之有关indexOf优化
Jan 15 #Javascript
用js小类库获取浏览器的高度和宽度信息
Jan 15 #Javascript
You might like
php单例模式实现(对象只被创建一次)
2012/12/05 PHP
php数组去重实例及分析
2013/11/26 PHP
php检测iis环境是否支持htaccess的方法
2014/02/18 PHP
ThinkPHP实现将SESSION存入MYSQL的方法
2014/07/22 PHP
PHP过滤黑名单关键字的方法
2014/12/01 PHP
php+html5基于websocket实现聊天室的方法
2015/07/17 PHP
thinkPHP实现递归循环栏目并按照树形结构无限极输出的方法
2016/05/19 PHP
thinkphp5使html5实现动态跳转的例子
2019/10/16 PHP
JS批量修改PS中图层名称的方法
2014/01/26 Javascript
提升PHP安全:8个必须修改的PHP默认配置
2014/11/17 Javascript
jQuery插件boxScroll实现图片轮播特效
2015/07/14 Javascript
微信小程序 location API实例详解
2016/10/02 Javascript
jQuery.cookie.js实现记录最近浏览过的商品功能示例
2017/01/23 Javascript
JS文件/图片从电脑里面拖拽到浏览器上传文件/图片
2017/03/08 Javascript
单行 JS 实现移动端金钱格式的输入规则
2017/05/22 Javascript
使用JQ完成表格隔行换色的简单实例
2017/08/25 Javascript
学习React中ref的两个demo示例
2018/08/14 Javascript
Vue CLI4 Vue.config.js标准配置(最全注释)
2020/06/05 Javascript
Element InputNumber计数器的使用方法
2020/07/27 Javascript
Python语法快速入门指南
2015/10/12 Python
Python实现简单的获取图片爬虫功能示例
2017/07/12 Python
用python生成1000个txt文件的方法
2018/10/25 Python
微信小程序python用户认证的实现
2019/07/29 Python
keras-siamese用自己的数据集实现详解
2020/06/10 Python
python中子类与父类的关系基础知识点
2021/02/02 Python
Html5 滚动穿透的方法
2019/05/13 HTML / CSS
马来西亚最大的电器网站:Senheng
2017/10/13 全球购物
表达自我的市场:Society6
2018/08/01 全球购物
匡威德国官网:Converse德国
2019/01/26 全球购物
面向游戏玩家和书呆子的极客订阅盒:Loot Crate
2020/11/25 全球购物
个人应聘自我评价分享
2013/11/18 职场文书
《和我们一样享受春天》教学反思
2014/02/07 职场文书
党员一句话承诺大全
2014/03/28 职场文书
《月亮湾》教学反思
2014/04/14 职场文书
教师听课学习心得体会
2016/01/15 职场文书
2019通用版新员工入职培训方案!
2019/07/11 职场文书