原型模式和基于原型继承的js对象系统
像同样基于原型编程的Io
语言一样,javascript
在原型继承方面,实现原理和Io非常类似,javascript也遵守这些原则
所有数据都是对象
要得到一个对象,不是通过实例化类,而是找到一个对象作为原型并克隆它
对象会记住它的原型
如果对象无法响应某个请求,它会把这个请求委托给它自己的原型.
一 , 所有数据都是对象
javascript
模仿了java
引入两套类型机制:基本型和对象类型.
基本型有:undefined
,number
,string
,Boolean
,function
,object
.
javascript中的根对象是object.prototype
,它是一个空的对象.javascript中的每个对象都是克隆自它.object.prototype
是他们的原型
var obj1 = new Object();
var obj2 = {};
console.log( Object.getPrototypeOf( obj1 ) === Object.prototype ); // 输出:true
console.log( Object.getPrototypeOf( obj2 ) === Object.prototype ); // 输出:true
二 , 要得到一个对象,不是通过实例化,而是找到一个对象作为原型并克隆它.
运用new
运算符从构造器中得到一个对象,比如:
function Person(name){
this.name = name;
}
person.prototype.getName = function (){
return this.name;
}
var a = new Person('cat');
console.log(a.name);//cat
console.log(a.getName() );//cat
console.log(object.getPrototypeOf(a) === Person.prototype);//true
Person并不是类,而是构造器.js的函数可以作为普通函数被调用,也可以作为构造器被调用.当使用new
运算符来调用函数时,此时函数就是一个构造器.
在Chrome和Firefox等向外暴露了对象_proto_
属性的浏览器下,我们可以通过以下代码来理解new
运算的过程
function Person(name){
this.name = name;
}
person.prototype.getName = function (){
return this.name;
}
var objectFactory = function(){
var obj = new Object(), //从Object.prototype上克隆一个空对象
Constructor = [].shift.call(arguments);//取得外部传入的构造器,此例是Person
obj._proto_ = Constructor.prototype; //指向正确的原型
var ret = Constructor.apply( obj, arguments );// 借用外部传入的构造器给 obj 设置属性
return typeof ret === 'object' ? ret : obj;// 确保构造器总是会返回一个对象
};
var a = objectFactory( Person, 'sven' );
console.log( a.name ); // 输出:sven
console.log( a.getName() ); // 输出:sven
console.log( Object.getPrototypeOf( a ) === Person.prototype );// 输出:true
三 , 对象会记住它的原型
对js来说,不能说对象有原型,只能说对象的构造器有原型."对象把请求委托给它自己的原型"这句话,更好的说法是对象把请求委托给它的构造器的原型.
js给对象提供了_proto_
的隐藏属性,这个属性指向它的构造器的原型对象,即{constructor}.prototype.
var a = new Object();
console.log(a._proto_ === object.prototype);//true
实际上,_proto_
就是对象与"对象构造器的原型"联系起来的纽带.
obj._proto_ = Constructor.prototype;
//通过这种方式来改变默认指向的object.prototype
四 , 如果对象无法响应某个请求,它会把这个请求委托给它的构造器原型
js中对象最初都是由Object.prototype
对象克隆而来,但是构造器的原型不仅限于object.prototype
,而是可以动态指向其他对象的.利用这一特性,
当对象a需要借用b的能力时,可以让a的构造器原型指向对象b,从而达到继承效果.
第一个例子:
var obj = {name:'nike'};
var A = function(){};
A.prototype = obj;
var a = new A();
console.log(a.name);//nike
这段代码运行的时候,引擎是这样工作的:
- 首先,尝试遍历对象
a
中所有的属性,但是没有找到name
这个属性. - 查找
name
属性的这个请求被委托给了对象a
的构造器的原型,它被a._proto_
记录下来并指向A.prototype
,而后者刚好被设置为对象obj. - 在对象
obj
中找到了name属性,并返回它的值.
第二个例子:
var A = function(){};
A.prototype = {name:'nike'};
var B = function(){};
B.prototype = new A();
var b = new B();
console.log(b.name);//nike
读这段代码,引擎是这样工作的:
- 首先,遍历对象
b
的属性,但是没有找到name
这个属性; - 查找任务被委托给
b
的构造器的原型B.prototype
,但是还是没有; - 由于
B.prototype
是通过A
构造器构造的对象,所以这个任务被委托给构造器A的原型A.prototype
; - 在
A.prototype
中找到name属性,返回值.
原型模式和基于原型继承的js对象系统的更多相关文章
- 面向对象的JavaScript --- 原型模式和基于原型继承的JavaScript对象系统
面向对象的JavaScript --- 原型模式和基于原型继承的JavaScript对象系统 原型模式和基于原型继承的JavaScript对象系统 在 Brendan Eich 为 JavaScrip ...
- 【原型模式】--重写原型对象prototype的影响
//[原型模式]--重写原型对象prototype的影响 2014-12-12//定义构造函数function Person() { }//直接指定构造函数的原型为一个对象(为了简化逐个给原型添加成员 ...
- 【js基础】创建对象的几种常见模式(工厂模式,构造函数模式,原型模式,构造原型组合模式)
一.工厂模式 缺点:没有解决对象识别的问题 优点:解决了创建多个相似对象的问题 function createPerson(name,age,job){ var o = new Object(); o ...
- 通过JavaScript原型链理解基于原型的编程
零.此文动机 用了一段时间的Lua,用惯了Java C++等有Class关键字的语言,一直对Lua的中的面向对象技术感到费解,一个开源的objectlua更是看了n遍也没理解其中的原理,直到看到了Pr ...
- JS构造函数、对象工厂、原型模式
1.对象创建的3中方法 1.1.对象字面量 var obj = { name: "mingzi", work: function () { console.log("wo ...
- js设计模式:工厂模式、构造函数模式、原型模式、混合模式
一.js面向对象程序 var o1 = new Object(); o1.name = "宾宾"; o1.sex = "男"; o1.a ...
- JS 设计模式一 -- 原型模式
原型模式 概念: 原型模式 是指原型实例指向创建对象的种类,并通过拷贝这些原型创建新的对象,是一种用来创建对象的模式,也就是创建一个对象作为另一个对象的prototype属性. 实现原型模式: 方法一 ...
- 原型模式 -- JavaScript语言的灵魂
原型模式就是将原型对象指向创建对象的类,使这些类共享原型对象的方法与属性.JS是基于原型链实现对象之间的继承,是对属性或者方法的共享,而不是对属性和方法的复制. // 图片轮播类 var LoopIm ...
- Java原型模式
原型模式 原型模式也称克隆模式.原型模式jian ming zhi yi,就是先创造出一个原型,然后通过类似于Java中的clone方法,对对象的拷贝,克隆类似于new,但是不同于new.new创造出 ...
随机推荐
- Andrew Ng机器学习课程笔记--week10(优化梯度下降)
本周主要介绍了梯度下降算法运用到大数据时的优化方法. 一.内容概要 Gradient Descent with Large Datasets Stochastic Gradient Descent M ...
- Python爬取糗事百科
import urllib import urllib.request from bs4 import BeautifulSoup """ 1.抓取糗事百科所有纯 ...
- 使用build_opener 自定义 opener
使用build_opener 自定义 opener,这种方法的好处是可以方便的拓展功能. import urllib.request import http.cookiejar def makeMyO ...
- hdu 6107--Typesetting(倍增)
题目链接 Problem Description Yellowstar is writing an article that contains N words and 1 picture, and t ...
- ssh框架知识点回顾
=========================================================================================== ======== ...
- SSH:Action中Service无法实例化
原来的代码: MailAction: public class MailAction extends ActionSupport { private SysuserinfoService sysuse ...
- linux(十二)之用户管理
前面学习了那么多关于linux的东西,相信大家都对linux应该 有一个大概的了解了.现在给大家分享的是linux中的用户管理,接下来让我们进入正题吧! 今天其实放松了一整天了,有点后悔自己没有把这些 ...
- python基础教程(二)
继续第一篇的内容,讲解,python的一些基本的东西. 注释 为了让别人能够更容易理解程序,使用注释是非常有效的,即使是自己回头再看旧代码也是一样. >>> #获得用户名: > ...
- js、JSP、servlet之间的传递小结
@ JS 与 JSP :JSP无法直接获取JS的值,只能通过隐藏表单或者dom节点设置. JSP中设置隐藏表单input,或者设置任意一个隐藏或者不隐藏的节点比如div, 而JS就通过document ...
- Python验证码通过pytesser识别
Python安装包: 需要安装的包主要有两个: PIL 和 pytesser .tesseract (1).安装PIL:下载地址:http://www.pythonware.com/products/ ...