一. 无中生有函数
起初,什么都没有。this
造物主说:没有东西自己也是一种东西啊,因而就有了null:prototype
如今咱们要造点儿东西出来。可是没有原料怎么办?对象
有一个声音说:不是有null嘛?blog
另外一个声音说:但是null表明无啊。继承
造物主说:那就无中生有吧!队列
因而:ip
JavaScript中的1号对象产生了,咱们把它叫作No. 1。原型
这个No. 1对象可不得了,它是真正的万物之祖。它拥有的性质和能力,是全部的对象都有的。自动化
__proto__是什么意思呢?那是“生”的意思,或者叫作“继承”。
既然有了一个对象,那么剩下就好办了,由于一辈子二,二生三,三生万物嘛。
可是,造物主很懒,他不想一个一个地亲手制造对象。因而他制造了一台可以造对象的机器:
他给这台机器起了一个名字:Object。
这台机器并不能凭空造出对象,它须要一个模板对象,按照这个模板对象来制造对象。很天然的,它把目前仅有的No.1对象做为模板。图中的prototype就表明机器的模板对象。
机器又叫作构造函数,为啥呢?由于它是用来构造对象的嘛。
机器如何启动呢?答案是经过new命令。你对着机器喊一声:“new!”,对象就造出来了。
机器的产生,使得生产对象的过程自动化了,解放了造物主的双手,因而造物主忙别的去了。
若是机器只是按照模板的样子,机械地复制出如出一辙的对象,那就太笨了。
人类的后代在继承了父辈的性状的基础上,能够产生父辈没有的新的性状。一样地,机器在制造对象时,除了继承模板对象的属性外,还能够添加新的属性。
好比说,有一天Object机器制造一个对象,它有一个特殊的属性,叫作flag,属性值是10。看起来是这样的:
写成代码就是:
var obj = new Object({ flag: 10 });
一每天过去了,造物主来视察工做,发现Object制造出了不少不少对象。他还注意到:根据“物以类聚”的原则,这些对象能够分红不少类。聪明的造物主想,我何很少造几台机器,让每一台机器负责制造一类对象呢?因而,他造出了几台机器并给它们起了名字,分别是:
String:用来制造表示一段文本的对象。
Number:用来制造表示一个数字的对象。
Boolean:用来制造表示是与非的对象。
Array:用来制造有序队列对象。
Date:用来制造表示一个日期的对象。
Error:用来制造表示一个错误的对象。
多台机器齐开动,各自制造本身负责的那一类对象。轰轰烈烈的造物运动开始了。
造物主又开始思考了,虽然机器是用来制造对象的,可是机器自己实际上也是一种特殊对象啊。如今有了这么多机器,我得好好总结一下它们的共同特征,把它们也归入对象体系。
因而,造物主基于No. 1对象,造出了一个No. 2对象,这个对象用来表示全部机器的共同特质。换句话说,它是全部机器的原型对象。
(注:__proto__写起来太麻烦了,咱们用[p]来代替。)
固然,同Object同样,这些机器也须要各自有一个模板对象,即它们的prototype属性指向的那个对象。显然,它们的模板对象应该是继承在No. 1对象的。即
这张图显示了JavaScript世界中那些最基本的机器自己的继承(__proto__)线路以及它们的模板对象的继承(prototype)线路。只是看起来太复杂了,因此后面咱们再也不把它们的prototype画出来。
造物主想:这下好了,我造出了Object机器,知足了对象制造的自动化。而后又造出了String、Number等机器,实现了特定类别的对象制造的自动化。可是,彷佛还缺点什么啊?
对了,还缺乏一台制造机器的机器啊!很快,万能的造物主就把它造了出来,并把它命名为Function。有了Function机器后,就能够实现自动化地制造机器了。
首先,Function也是一台机器,因此它的原型对象也是No. 2对象。
其次,Function又是一台制造机器的机器,因此它的模板对象也是No. 2对象。
因此咱们获得了Function的一个很是特别的性质:
Function.__proto__ === Function.prototype
哇,太奇妙了!
不要奇怪,这个性质不过是”Function是一台制造机器的机器“这个事实的必然结果。
因而JavaScript的世界的变成了这个样子:
从这张图上,咱们会发现:全部的函数(包括Function)的__proto__都指向No.2 对象,而同时Function.prototype也是No.2 对象。这说明了:
从逻辑上,咱们能够认为全部机器(包括Function本身)都是由Function制造出来的。
同时,若是再仔细瞧瞧,你会发现:
Object做为一个机器能够看作是有由Function制造出来的,而Function做为一个对象能够看作是由Object制造出来的。
这就是JavaScript世界的“鸡生蛋,蛋生鸡”问题。那么究竟是谁生了谁呢?Whatever!
根据上文的叙述,机器用来制造某一类对象。正由于如此,机器能够做为这类对象的标志,即面向对象语言中类(class)的概念。此时,机器被称为构造函数。因此,在ES6引入class关键字以前,咱们经常把构造函数叫作类。
然而,除了做为构造函数来制造对象外,函数一般还用做另外的用途:用来作一件事情。正是有了这个功能,JavaScript的世界才由静变更,变得生机勃勃。
好比,咱们如今用Function机器制造了鸟类(即用来造鸟的机器):
function Bird(color) { this.color = color; }
而后,对着造鸟机说:“new!”,因而造鸟机发动起来,制造一个红色的鸟:
var redBird = new Bird('#FF0000');
如今咱们想让鸟飞起来,因而咱们再用Function机器来制造一台机器。这台机器不是用来制造对象的,而是用来作一件事情的,即“让鸟飞起来”这件事情:
// 这是一台经过晃动鸟的翅膀,让鸟飞起来的简陋的机器。 function makeBirdFly(bird) { shakeBirdWing(bird); }
咱们知道,让一台制造对象的机器发动,只须要对它喊“new”便可;那么怎样让一台作事情的机器发动呢?更简单,对它咳嗽一声就好了。
makeBirdFly(redBird);
因而红鸟飞起来了,世界充满了生机。
从上面的Bird和makeBirdFly的定义能够看出:实际上,制造对象的机器和作事情的机器没什么明显区别,它们只是使用方式不一样。在两种状况下,它们分别被叫作构造函数和普通函数。
说明1:function xxx语法能够当作new Function的等价形式。
说明2:用户自定义的函数一般既能够做为普通函数使用,又能够做为构造函数来制造对象。
ES6新增的class语法定义的函数只能做为构造函数,ES6新增的=>语法定义的箭头函数只能做为普通函数。
造物主对目前的世界不太满意。由于几乎全部的机器的模板对象都是No. 2,这致使世界看起来有点扁。
因而他又开始研究世界万物的分类问题。它发现有些对象会动、还会吃东西,因而他把它们叫作动物,用机器Animal来制造它们。他进一步发现,即便都是动物,也仍是能够进一步分类,好比有些会飞、有些会游,他分别把它们叫作鸟类、鱼类。因而他想,我何不单独造几台机器,专门用来制造某一类动物呢。因而它造出了Bird、Fish等机器。
接下来,在选择这些机器的模板对象时碰到一个问题:若是还像以前那样直接复制一个No. 1对象做为Bird、Fish的模板,那么结果就是这样的:
这样可很差。首先没体现出鸟类、鱼类跟动物的关系,其次它们的模板对象存了重复的东西,这是一种浪费。怎么办呢?很简单,让Bird和Fish的模板对象继承自Animal的模板对象就行了。也就是
Bird.prototype.__proto__ === Animal.prototype Fish.prototype.__proto__ === Animal.prototype
因而:
用一样的方法,造物主造出了一个立体得多的JavaScript世界。
然而还不够。虽然那些纯对象如今充满了层次感,可是那些机器之间的关系仍是扁平的:
怎么办呢?其实用相似的办法就好了:
为了作到这点,造物主发明了class关键字。
世界如今变得可复杂了,只能画出一部分: