和java这种基于类(class-base)的面向对象的编程语言不同,javascript没有类这样的概念,但是javascript也是面向对象的语言,这种面向对象的方式成为 基于原型(prototype-base)的面向对象。虽然说ES6已经引入了类的概念来作为模板,通过关键字 “class” 可以定义类,但ES6的这种写法可以理解为一种语法糖,它的绝大部分功能,ES5都可以做到,新的class写法只是让对象原型的写法更加清晰、更像面向对象编程的语法而已。如果要理解基于原型实现面向对象的思想,那么理解javascript中得三个重要概念: 构造函数(constructor)、原型(prototype)、原型链(prototype chain) 对帮助理解基于原型的面向对象思想就显得尤为重要。下面就重点介绍一下这几个概念。
先来看一张来自 mollypages.org 的javascript对象的结构图,下面的例子都按照这张图阐述。
构造函数是用来初始化对象的,每个构造函数都有一个不可枚举的属性,这就是原型(prototype).并且,每个prototype 都包含一个包含了不可枚举属性的constructor属性,这个属性始终会指向构造函数。
javascript
function Foo(){}; console.log(Foo.prototype.constructor === Foo); // true
每个被new实例化的对象都会包含一个 __proto__
属性,它是对构造函数 prototype
的引用。
javascript
function Foo(){}; var foo = new Foo(); console.log(foo.__proto__ === Foo.prototype); // ture
javascript
function Foo(){}; console.log(Foo.prototype.__proto__ === Object.prototype); // true
上面返回true 的原因是Foo.prototype 是Object预创建的一个对象,是Object创建的一个实例,所以,Foo.prototype.__proto_ 是Object.prototype的引用。
我们可以来看一下原型链的脉络。
在javascript中,函数是一种特殊的对象,所有的函数都是构造函数 Function
的实例。
javascript
function Foo() {}; console.log(Foo.__proto__ === Object.prototype); //false console.log(Fool.__proto__ === Function.prototype); // true
从上面可以看出,函数Foo.__proto_ 指向到 Function.prototype, 说明函数 Function是 Function的一个实例。
javascript
function Foo(){}; console.log(Foo.__proto__ === Function.prototype); //true console.log(Foo.prototype.__proto__ === Object.prototype);//true
Foo.prototype 是Object预定义的对象,构造函数为Object所以 proto 指向 Object.proto.
从上面的图我们可以看出, Object、Function、Array 等这些函数,他们的构造函数都是 Function 的实例。
有了上面的基础知识以后,我们就可以自己去基于原型链去封装对象,实现javascript的继承。先来看下面一个例子。
运行上面的例子,输入 cat init 和 animal eat,说明cat 继承了 Animal.prototype.eat 的方法。
我们来分析一下代码。
1、Animal 的prototype中定义了 eat方法。
2、将Empty.prototype 指向 Animal.prototype , 所以 Empty.prototype 中也存在 eat
方法.
3、Cat.prototype == new Empty(),所以 Cat.prototype.__proto_ === Animal.__proto_.
4、重新为Cat指定constructor为Cat,否则Cat不存在constructor。
这样就完成了继承,原型链是这样的:
这样我们用原型链的方式实现了一个简单的继承方式。
基于原型的javascript 面向对象编程