JavaScript 隐式原型(_proto_)与显示原型(prototype)

做者:苏墨橘
连接:https://www.zhihu.com/question/34183746/answer/59043879
来源:知乎
著做权归做者全部。商业转载请联系做者得到受权,非商业转载请注明出处。

react

正好这段时间在从新看这部分,写一篇回答来梳理一下吧。web

__proto__(隐式原型)与prototype(显式原型)express

1. 是什么
  • 显式原型 explicit prototype property:
每个函数在建立以后都会拥有一个名为prototype的属性,这个属性指向函数的原型对象。
Note:经过Function.prototype.bind方法构造出来的函数是个例外,它没有prototype属性。(感谢 同窗的答案让我知道这一点)
NOTE Function objects created using Function.prototype.bind do not have a prototype property or the [[Code]], [[FormalParameters]], and [[Scope]] internal properties. ----- ECMAScript Language Specification
  • 隐式原型 implicit prototype link:
JavaScript中任意对象都有一个内置属性[[prototype]],在ES5以前没有标准的方法访问这个内置属性,可是大多数浏览器都支持经过__proto__来访问。ES5中有了对于这个内置属性标准的Get方法Object.getPrototypeOf().
Note: Object.prototype 这个对象是个例外,它的__proto__值为null
  • 两者的关系:

隐式原型指向建立这个对象的函数(constructor)的prototype浏览器

2. 做用是什么
  • 显式原型的做用:用来实现基于原型的继承与属性的共享。
ECMAScript does not use classes such as those in C++, Smalltalk, or Java. Instead objects may be created in various ways including via a literal notation or via constructors which create objects and then execute code that initialises all or part of them by assigning initial values to their properties. Each constructor is a function that has a property named “prototype” that is used to implement prototype-based inheritance and shared properties.Objects are created by using constructors in new expressions; for example, new Date(2009,11) creates a new Date object. ----ECMAScript Language Specification
  • 隐式原型的做用:构成原型链,一样用于实现基于原型的继承。举个例子,当咱们访问obj这个对象中的x属性时,若是在obj中找不到,那么就会沿着__proto__依次查找。
Every object created by a constructor has an implicit reference (called the object’s prototype) to the value of its constructor’s “prototype” ----ECMAScript Language Specification
3. __proto__的指向
__proto__的指向到底如何判断呢?根据ECMA定义 'to the value of its constructor’s "prototype" ' ----指向建立这个对象的函数的显式原型。因此关键的点在于找到建立这个对象的构造函数,接下来就来看一下JS中对象被建立的方式,一眼看过去彷佛有三种方式:(1)对象字面量的方式 (2)new 的方式 (3)ES5中的Object.create() 可是我认为本质上只有一种方式,也就是经过new来建立。为何这么说呢,首先字面量的方式是一种为了开发人员更方便建立对象的一个语法糖,本质就是 var o = new Object(); o.xx = xx;o.yy=yy; 再来看看Object.create(),这是ES5中新增的方法,在这以前这被称为原型式继承,
道格拉斯在2006年写了一篇文章,题为 Prototypal Inheritance In JavaScript。在这篇文章中,他介绍了一种实现继承的方法,这种方法并无使用严格意义上的构造函数。他的想法是借助原型能够基于已有的对象建立新对象,同时还不比所以建立自定义类型,为了达到这个目的,他给出了以下函数:
function object(o){
    function F(){}
    F.prototype = o;
    return new F()
}
----- 《JavaScript高级程序设计》P169
因此从实现代码 return new F() 中咱们能够看到,这依然是经过new来建立的。不一样之处在于由 Object.create() 建立出来的对象没有构造函数,看到这里你是否是要问,没有构造函数我怎么知道它的__proto__指向哪里呢,其实这里说它没有构造函数是指在 Object.create() 函数外部咱们不能访问到它的构造函数,然而在函数内部实现中是有的,它短暂地存在了那么一下子。假设咱们如今就在函数内部,能够看到对象的构造函数是F, 如今
//如下是用于验证的伪代码
var f = new F(); 
//因而有
f.__proto__ === F.prototype //true
//又由于
F.prototype === o;//true
//因此
f.__proto__ === o;

所以由Object.create(o)建立出来的对象它的隐式原型指向o。好了,对象的建立方式分析完了,如今你应该可以判断一个对象的__proto__指向谁了。函数

好吧,仍是举一些一眼看过去比较疑惑的例子来巩固一下。ui

  • 构造函数的显示原型的隐式原型:
  1. 内建对象(built-in object):好比Array(),Array.prototype.__proto__指向什么?Array.prototype也是一个对象,对象就是由 Object() 这个构造函数建立的,所以Array.prototype.__proto__ === Object.prototype //true,或者也能够这么理解,全部的内建对象都是由Object()建立而来。
  • 自定义对象
1. 默认状况下:
function Foo(){}
var foo = new Foo()
Foo.prototype.__proto__ === Object.prototype //true 理由同上
2. 其余状况:
(1)
 function Bar(){}
//这时咱们想让Foo继承Bar
Foo.prototype = new Bar()
 Foo.prototype.__proto__ === Bar.prototype //true
(2)
//咱们不想让Foo继承谁,可是咱们要本身从新定义Foo.prototype 
Foo.prototype = {
  a:10,
  b:-10
}
//这种方式就是用了对象字面量的方式来建立一个对象,根据前文所述 
Foo.prototype.__proto__ === Object.prototype

: 以上两种状况都等于彻底重写了Foo.prototype,因此Foo.prototype.constructor也跟着改变了,因而乎constructor这个属性和原来的构造函数Foo()也就切断了联系。spa

  • 构造函数的隐式原型

既然是构造函数那么它就是Function()的实例,所以也就指向Function.prototype,好比 Object.__proto__ === Function.prototypeprototype

4. instanceof
instanceof 操做符的内部实现机制和隐式原型、显式原型有直接的关系。instanceof的左值通常是一个对象,右值通常是一个构造函数,用来判断左值是不是右值的实例。它的内部实现原理是这样的:
//设 L instanceof R 
//经过判断
 L.__proto__.__proto__ ..... === R.prototype ?
//最终返回true or false
也就是沿着L的__proto__一直寻找到原型链末端,直到等于R.prototype为止。知道了这个也就知道为何如下这些奇怪的表达式为何会获得相应的值了
 Function instanceof Object // true 
 Object instanceof Function // true 
 Function instanceof Function //true
 Object instanceof Object // true
 Number instanceof Number //false

文章参考:JavaScript instanceof 运算符深刻剖析设计

相关文章
相关标签/搜索