JavaScript 中的原型对象和原型链
on Front-End
普通对象与函数对象
当然这里是说除了 undefined
、null
、boolean
、number
、string
这 5 种按值访问的基本数据类型(也叫值类型)。剩下的都是按引用访问的对象类型(也叫引用类型)。值类型的类型判断用 typeof
,引用类型的类型判断用 instanceof
。
对象的定义很简单,说白了,对象就是属性的集合。比如,函数是对象的一种,它也有自己的属性。例如,在 jQuery
源码中,jQuery
或者 $
这个变量其实是一个函数,而 $.trim()
也是一个函数,所以这就可以说明 trim 其实就是 $
的一个属性名,这个属性名对应的属性值是一个函数。
基于上述的例子,侧面印证了这样一个结论:一切引用类型都是对象,而对象是属性的集合。但对象也是有区别的,分为普通对象和函数对象。那么又怎么区分普通对象和函数对象呢?
凡是通过 new Function()
创建的对象都是函数对象,其他的都是普通对象。
函数和对象的关系
先看一个例子:
function Blog() {
this.name = 'duncantim21';
this.year = 2016;
}
var blog = new Blog();
可以看到,对象是可以通过函数来创建的。但事实不止于此,因为:对象都是通过函数 new 出来的。
有人可能会反驳,这不对吧,下面的 obj
和 arr
明明不是通过函数创建的。
var obj = { a : 10, b : 20 };
var arr = [7, 'w', true];
但是不好意思,其实以上代码的本质是:
var obj = new Object();
obj.a = 10;
obj.b = 20;
var arr = new Array();
arr[0] = 7;
arr[1] = 'w';
arr[2] = true;
而其中的 Object
和 Array
都是函数。
至此,我们发现,对象是由函数创建的,而函数却又是一种对象。它不像数组一样简单的只是对象的子集,函数和对象的关系简直就是鸡生了蛋,蛋又生了鸡。
显式原型对象 prototype
前面说了函数是对象,对象就有属性。JavaScript
就给每个函数对象都设置了一个默认的属性:prototype
,这个属性指向函数的原型对象。同时,这个 prototype
的属性值也是一个对象,这个对象还有一个默认的属性:constructor
,指向这个函数本身,即 Blog.prototype.constructor === Blog
。另外,一个实例的 constructor
属性指向其造函数,即 blog.constructor === Blog
,Object.constructor == Function
。
由此可以看出,原型对象首先就是一个普通对象,同时它是构造函数的一个实例。
既然这个 prototype
是一个对象,那我们就可以自定义增加许多属性。通过自定义属性,那么由一个函数创建的对象就拥有了这个函数原型 prototype
上的的所有属性。
同时,JavaScript
还为每个对象都设置了一个默认的隐藏属性 __proto__
,这个属性指向创建该对象的函数的 prototype
。即:blog.__proto__ === Blog.prototype
。
隐式原型对象 __proto__
既然函数的 prototype
也是对象,这个对象本质上就和 var obj = {}
一样都是被 Object
创建的,所以它的 __proto__
指向的就是 Object.prototype
。
另一方面由于对象都是由函数创建的,也就是说包括自定义函数在内的所有对象都是由函数 Function
创建的,所以所有函数对象的 __proto__
都指向 Function.prototype
,它是一个空函数,即 Object.__proto__ === Function.prototype
。
同时,Function
也是一个函数和一种对象,所以 Function
是被自身创建的,即 Function.__proto__ === Function.prototype
,Function.prototype.__proto__ === Object.prototype
。
运算符 instanceof
instanceof
运算符其实表示的是一种继承关系,或者原型链的结构。它的第一个变量是一个对象,暂时称为 A;第二个变量一般是一个函数,暂时称为 B。它的判断规则是:同时沿着 A 的 __proto__
和 B 的 prototype
来找,如果两条线能找到同一个引用对象,那么就返回 true
。如果找到终点还未重合,则返回 false
。
以下是原型链结构的图示说明,可以反复对照方便理解和记忆:
原型链
访问一个对象的属性时,先在其基本属性中查找,如果没有,再沿着 __proto__
这条链向上找,这就是原型链。
那么我们在实际应用中如何区分一个属性到底是基本的还是从原型中找到的呢?答案就是:hasOwnProperty
,特别是在 for…in…
循环中一定要注意。
这个 hasOwnProperty
函数是从 Object.prototype
中来的。由于所有对象的原型链都会找到 Object.prototype
,因此所有对象都会有 Object.prototype
里的函数。这就是所谓的”继承”。
一些例外
typeof null // object
typeof function() {} // function
typeof Object // function
typeof Function.prototype // function
typeof Function.prototype.prototype // undefined
Object.prototype.__proto__ === null // true
Math.__proto__ === Object.prototype // true
Math.construrctor === Object // true
JSON.__proto__ === Object.prototype // true
JSON.construrctor === Object //true