详解JavaScript中的__proto__与prototype(上)

  |  
 阅读次数

先看一段代码

1
2
3
4
5
Function instanceof Object // true
Object instanceof Function // true
Function instanceof Function //true
Object instanceof Object // true
Number instanceof Number //false

再来看一段

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
var obj = { a : 1 };
console.log(obj.__proto__ === Object.prototype); // true
var str = new String('123');
console.log(str.__proto__ === String.prototype); // true
function Point(){};
var Circle = Object.create(Point);
console.log(Circle.__proto__ === Point); // true
console.log(Circle.__proto__ === Point.prototype); // false
var p = new Point();
console.log(Point.__proto__); // function()
console.log(Point.prototype); // Point {}
console.log(p.__proto__); // Point {}
console.log(p.prototype); // undefined

很明显,__proto__与prototype看起来似乎很相似,但是实际上是不同的。
先了解一下各自是什么,再进一步探讨。

prototype 显示原型

每一个函数在创建之后都会拥有一个名为prototype的属性,这个属性指向函数的原型对象。(通过Function.prototype.bind方法构造出来的函数是个例外,它没有prototype属性。)

__proto__ 隐式原型

JavaScript中任意对象都有一个内置属性[[prototype]],在ES5之前没有标准的方法访问这个内置属性,但是大多数浏览器都支持通过proto来访问。ES5中有了对于这个内置属性标准的Get方法Object.getPrototypeOf().
Note: Object.prototype 这个对象是个例外,它的__proto__值为null

二者关系

隐式原型指向创建这个对象的函数(constructor)的prototype

1
2
3
function test(){};
var a = new test();
console.log(a.__proto__ === test.prototype) ;// true

梳理

JavaScript里万物皆对象,方法(Function)是一种特殊的对象。
由上述的定义可知,任意对象都有都有__proto__,而prototype只有函数创建之后自己才有,通过该函数创建的对象是没有的。

1
2
3
4
function test(){};
var a = new test();
console.log(test.prototype); // test{}
console.log(a.prototype); // undefined

而Object.prototype这个特殊的对象的__proto__位于原型链金字塔的顶端,为null,如下图
图形化原型链
这里值得一提的是JavaScript的原型链中原型对象prototype之间是通过__proto__联系起来的
__proto__
同时原型对象prototype中都有个预定义的constructor属性,用来引用它的函数对象。这是一种循环引用

1
2
3
person.prototype.constructor === person //true
Function.prototype.constructor === Function //true
Object.prototype.constructor === Object //true

这一点在本文后面的图中也有显示,Click

有两点需要注意:

(1)注意Object.constructor===Function;//true 本身Object就是Function函数构造出来的
(2)如何查找一个对象的constructor,就是在该对象的原型链上寻找碰到的第一个constructor属性所指向的对象

再看上面一截代码中的部分

1
2
3
function Point(){};
var Circle = Object.create(Point);
console.log(Circle.__proto__ === Point); // true

Object.create(),这是ES5中新增的方法,在这之前这被称为原型式继承,我们可以理解为 new Object(),这样我们就很好理解结果true了,实际上,上述代码可以等价理解为

1
2
3
function Point(){};
var Circle = new Point();
console.log(Circle.__proto__ === Point); // true

等价的原因在上面二者关系有提到。
也可以从constructor来理解

1
2
console.log(Circle.constructor); //function Point()
console.log(Circle.constructor.prototype === Circle.__prototype) // true

instanceof

instanceof 运算符用来测试一个对象在其原型链中是否存在一个构造函数的 prototype 属性。

语法

1
object instanceof constructor

参数

object
要检测的对象.

constructor
某个构造函数

描述

instanceof 运算符用来检测 constructor.prototype 是否存在于参数 object 的原型链上。

1
2
3
4
//设 L instanceof R
//通过判断
L.__proto__.__proto__ ..... === R.prototype ?
//最终返回true or false

也就是沿着L的__proto__一直寻找到原型链末端,直到等于R.prototype为止,如果一直到Object.prototyoe都没找到,则返回false。就此,我们结合下面这幅图再剖析一下文章开头部分的代码段就很容易理解了

__proto__与prototype

参照知乎知乎–js中proto和prototype的区别和关系?整理得出,加入部分个人见解,有错之处望各位大牛斧正