可以将原型理解为对象的父亲,对象从原型对象继承属性。
1.所有函数的原型默认是 Object的实例,所以可以使用toString/toValues/isPrototypeOf 等方法的原因。
data:image/s3,"s3://crabby-images/eedc7/eedc738504315b3e318826147a02658a5153d63c" alt=""
data:image/s3,"s3://crabby-images/a0e52/a0e52aa950e61dd9b449defbd31b21426011060d" alt=""
2.我们也可以创建一个极简对象(纯数据字典对象)没有原型(原型为null)
data:image/s3,"s3://crabby-images/2c1bf/2c1bf4aea48fe13b39e703cfd65a9316af579ed1" alt=""
3.函数拥有多个原型,prototype 用于实例对象使用,__proto__用于函数对象使用
data:image/s3,"s3://crabby-images/10645/106452f48995dbb927f3477554511789346f4632" alt=""
data:image/s3,"s3://crabby-images/8ae34/8ae3465bbd3c8c6b66e738e429376cc32a9d2c76" alt=""
4.使用Object.setPrototypeOf 可设置对象的原型。使用Object.getPrototypeOf 用于获取一个对象的原型。
data:image/s3,"s3://crabby-images/31435/31435604740ec6b1206fdf74e8a747c6e1721aec" alt=""
5.原型检测
一、instanceof 检测构造函数的 prototype 属性是否出现在某个实例对象的原型链上。
二、使用isPrototypeOf检测一个对象是否是另一个对象的原型链中,b.isPrototype(a)。
三、使用in 检测原型链上是否存在属性(“name” in a),使用 hasOwnProperty 只检测当前对象(a.hasOwnPrototype("name"))。
6.除了使用Object.setPrototypeOf(a,b)让a继承b的原型,从而使用b拥有的方法。使用call和apply借助其他原型方法来完成功能。
data:image/s3,"s3://crabby-images/2f313/2f313b4ff7e5f95209f84f367c54c82305cd448e" alt=""
也就是说我们可以通过这种方式直接借用String/Number/Math/RegExp/Date等系统方法了。
data:image/s3,"s3://crabby-images/dea27/dea27afdad42c3cf627d1a16097f15004f20676e" alt=""
7.使用Object.getPorotypeOf获得对象的原型,而constructor存在于prototype原型中,用于指向构建函数的引用。
data:image/s3,"s3://crabby-images/fe2c0/fe2c008e595124f587e31c0ac9ef5995cf02e079" alt=""
data:image/s3,"s3://crabby-images/b1c41/b1c4178327709f6b18450f7ff12b5c0a11678ec5" alt=""
8.给我一个对象,还你整个世界。通过一个对象可以找到原型(Object.getPorotype()),通过原型的constructor能够找到构造函数,通过构造函数就可以创建一个对象。
data:image/s3,"s3://crabby-images/23e44/23e4437acf60e290ffa2582619f988c17a2d1d39" alt=""
根据上面讲到的__proto__ 我们来分析下,首先obj是没有constructor 这个属性的,但是 obj.__proto__ = a.prototype;就从a.prototype中寻找,而 a.prototype.constructor 是就a,所有两者的结果是一一样的.
data:image/s3,"s3://crabby-images/09fdb/09fdbaaff0fc478195224e1cd5598c680119fea5" alt=""
通俗的讲,就是为了将实例的构造器的原型对象暴露出来, 比如你写了一个插件,别人得到的都是你实例化后的对象, 如果别人想扩展下对象,就可以用 instance.constructor.prototype 去修改或扩展原型对象。
9.使用Object.create创建一个新对象时使用现有对象做为新对象的原型对象。在实例化对象上存在 __proto__ 记录了原型,所以可以通过对象访问到原型的属性或方法。修改对象的__proto__只语序是对象或者是null
data:image/s3,"s3://crabby-images/a2f3c/a2f3cca72cb68b5b873fac2ffbb3afed251ccf6e" alt=""
data:image/s3,"s3://crabby-images/02d5b/02d5bcdd929f83ea32c97ddaf245f153ee991fd0" alt=""
10.使用Object.create()实现继承会影响对象constructor 。
如:Admin.prototype = Object.create(User.prototype);需要设置Admin.prototype.constructor = Admin;确保constructor属性存在。这使这个属性变成了可遍历的,所以使用Object.defineProperty(Admin.prototype,"constructor",{ value:Admin,enumerale:false})就不可以遍历了。可以将这两行代码封装成一个方法,减少重复书写类似代码。
11.javascript脚本语言不支持多继承,如果要使用多个类的方法时可以使用mixin混合模式来完成。
网友评论