JavaScript 原型中的哲学思想

 

学习JavaScript过程中,原型问题一直让我疑惑许久,那时候捧着那本著名的红皮书,看到有关原型的讲解时,总是心存疑虑。
  当在JavaScript世界中走过不少旅程之后,再次萌发起研究这部分知识的欲望,翻阅了不少书籍和资料,才搞懂__proto__和prototype的概念。

  故以作此笔记,日后忘了可以回来看看。

  如果你看的过程中觉得理解有些困难,把例子在代码中跑一跑,亲手试一试也许能解决不少疑惑。

  一切皆为对象
  殊不知,JavaScript的世界中的对象,追根溯源来自于一个 null



  「一切皆为对象」,这句着实是一手好营销,易记,易上口,印象深刻。


  万物初生时,一个null对象,凭空而生,接着Object、Function学着null的模样塑造了自己,并且它们彼此之间喜结连理, 提供了prototype和constructor,一个给子孙提供了基因,一个则制造万千子子孙孙。

  在JavaScript中,null也是作为一个对象存在,基于它继承的子子孙孙,当属对象。

  乍一看,null像是上帝,而Object和Function犹如JavaScript世界中的亚当与夏娃。

  原型指针 __proto__

  在JavaScript中,每个对象都拥有一个原型对象,而指向该原型对象的内部指针则是__proto__,通过它可以从中继承原型对象的属性,原型是JavaScript中的基因链接,有了这个,才能知道这个对象的祖祖辈辈。从对象中的__proto__可以访问到他所继承的原型对象。
  var a = new Array(); 
  a.__proto__ === Array.prototype // true
  上面代码中,创建了一个Array的实例a,该实例的原型指向了Array.prototype。

  Array.prototype本身也是一个对象,也有继承的原型:
  a.__proto__.__proto__ === Object.prototype // true 
  // 等同于 Array.prototype.__proto__ === Object.prototype
  这就说了明了,Array本身也是继承自Object的,那么Object的原型指向的是谁呢?
  a.__proto__.__proto__.__proto__ === null // true 
  // 等同于 Object.prototype.__proto__ === null


  所以说,JavaScript中的对象,追根溯源都是来自一个null对象。佛曰:万物皆空,善哉善哉。

  除了使用.__proto__方式访问对象的原型,还可以通过Object.getPrototypeOf方法来获取对象的原型,以及通过Object.setPrototypeOf方法来重写对象的原型。

  值得注意的是,按照语言标准,__proto__属性只有浏览器才需要部署,其他环境可以没有这个属性,而且前后的两根下划线,表示它本质是一个内部属性,不应该对使用者暴露。因此,应该尽量少用这个属性,而是用Object.getPrototypeof和Object.setPrototypeOf,进行原型对象的读写操作。


  这里用__proto__属性来描述对象中的原型,是因为这样来得更加形象,且容易理解。

  原型对象 prototype

  函数作为JavaScript中的一等公民,它既是函数又是对象,函数的原型指向的是Function.prototype
  var Foo = function() {} 
  Foo.__proto__ === Function.prototype // true


  函数实例除了拥有__proto__属性之外,还拥有prototype属性。
  通过该函数构造的新的实例对象,其原型指针__proto__会指向该函数的prototype属性。
  var a = new Foo(); 
  a.__proto__ === Foo.prototype; // true
  而函数的prototype属性,本身是一个由Object构造的实例对象。
  Foo.prototype.__proto__ === Object.prototype; // true
  prototype属性很特殊,它还有一个隐式的constructor,指向了构造函数本身。
  Foo.prototype.constructor === Foo; // true
  a.constructor === Foo; // true
  a.constructor === Foo.prototype.constructor; // true


  原型链

  概念

  原型链作为实现继承的主要方法,其基本思想是利用原型让一个引用类型继承另一个引用类型的属性和方法。

  每个构造函数都有一个原型对象(prototype),原型对象都包含一个指向构造函数的指针(constructor),而实例都包含一个指向原型对象的内部指针(__proto__)。

  那么,假如我们让原型对象等于另一个类型的实例,此时的原型对象将包含一个指向另一个原型的指针,相应地,另一个原型中也包含着一个指向另一个构造函数的指针。假如另一个原型又是另一个类型的实例,那么上述关系依然成立。

  如此层层递进,就构造了实例与原型的链条,这就是原型链的基本概念。

  意义

  “原型链”的作用在于,当读取对象的某个属性时,JavaScript引擎先寻找对象本身的属性,如果找不到,就到它的原型去找,如果还是找不到,就到原型的原型去找。以此类推,如果直到最顶层的Object.prototype还是找不到,则返回undefine

  亲子鉴定

  在JavaScript中,也存在鉴定亲子之间DNA关系的方法:

  1、instanceof
  运算符返回一个布尔值,表示一个对象是否由某个构造函数创建。

  2、Object.isPrototypeOf()
  只要某个对象处在原型链上,isProtypeOf都返回true
  var Bar = function() {} 
  var b = new Bar(); 
  b instanceof Bar // true 
  Bar.prototype.isPrototypeOf(b) // true 
  Object.prototype.isPrototypeOf(Bar) // true

要注意,实例b的原型是Bar.prototype而不是Bar

posted @ 2017-02-05 11:25  CC大神01  阅读(146)  评论(1编辑  收藏  举报