我们创建的每一个函数都有一个prototype(原型)属性,这个属性是一个指针,指向一对象,而这个对象的用途包括可以由特定类型的所有实例共享属性和方法。
简单来说,创建函数时浏览器会自动创建一个prototype属性指向原型对象,原型对象可以储存方法和属性由所有对象实例共享使用。
创建原型对象时,默认的也会获得一个constructor(构造函数)属性,这个属于又是指向 prototype 所在的一个指针。
当调用构造函数创建一个新实例后,该实例的内部将包含一个指针,指向构造函数的原型对象。在ECMA-262第5版管这个指针为 [[Prototype]] 。
虽然在脚本中没有标准的方式访问 [[Prototype]] ,但Firefox,Safari,Chrome在每个对象中都支持一个 proto 属性;要明确真正重要的是,这个连接存在于实例与构造函数的原型对象之间,而不是实例与构造函数之间。
isPrototypeOf()
[[Prototype]] 指向 (构造函数实例) 调用方法的对象(),那么这个方法就会返回true。
如下所示:
var Dome1 = new Dome();
Dome.prototype.isPrototypeOf(Dome1); //true
就是说Dome1的 [[Prototype]] 指向了Dome函数的原型对象就会返回true。
hasOwnProperty()
检测一个属性是否存在于实例中,还是原型中。存在于实例中会返回true,反之false。
构造对象.hasOwnProperty("属性")
in操作符
通过对象访问到目标属性是返回true,无论在实例还是原型中。
"属性" in 构造对像
如图,Persona.Prototype = New Man() 把Man函数的[[Prototype]]属性连接到Persona函数的原型对象中,相当于Persona继承了Man原型的属性和方法。
在对象实例请求使用某个属性和方法,就会在实例中查找,如果找不到就继续查找原型对象,甚至继续查找原型对象所继承的原型对像,形成的一种链型方式,就像连着链子一直寻找,而这个链子可以称为原型链。
通过原型链实现继承时,不能使用对象字面量创建和添加原型方法。因为这样会重写原型链,导致继承无效。
借用构造函数 (也叫伪造函数或经典继承)
这种技术的基本思想相当简单,既在子类型构造函数的内部调用超类型构造函数。函数只不过是在特定环境中执行代码的对象,因此通过使用apply()和call()方法也可以在 (将来) 新创建的对象上执行构造函数。