2006 年,Douglas Grockford 写了一篇文章:《JavaScript 中的原型式继承》(”Prototypal Inheritance in JavaScript“)。这篇文章介绍了一种不波及严格意义上构造函数的继承办法。他的出发点是即便不自定义类型也能够通过原型继承对象之间的信息共享。文章最终给出了一个函数:
function object(o) {function F(){} F.prototype = o return new F()}
于是乎,《JavaScript 高级程序设计》中的 JavaScript 就多了一种——原型式继承
于是乎,ECMAScript 5 新增了 Object.create() 办法将原型式继承的概念规范化
用法
var obj = Object.create({name: 'johan', age: 23}) // obj 继承了属性 name 和 age
var obj2 = Object.create(null) // obj2 不继承任何属性和办法
var obj3 = Object.create(Object.prototype) // 与 {} 和 new Object() 一个意思
var obj4 = Object.create({}, {
property1: {
value: true,
writable: true
}
}) // 第二个参数与 Object.defineProperties() 统一
图解 Object.create 实现
function create(proto) {function F(){}
F.prototype = proto
return new F()}
第一步: function F(){}
即创立一个函数,因为约定首字母大写,视为构造函数,创立函数 F 时,F 构造函数与和它的原型对象就有了这一层的关系:
F.prototype === F.prototype; // 假如你把 F.prototype 当作一个值
F.prototype.constructor === F;
第二步:F.prototype = proto
行将 F.prototype
赋值为传入的 proto,如此就突破了 F.prototype = F.prototype
以及 F.prototype.constructor = F
,它们的关系为
第三步:return new F()
第三步的解读有点费解,因为这里波及到 new 的操作,在 new 扭转了对象 中咱们说过,new 会创立一个对象,并将这个对象的隐式原型 (__proto__
) 指向构造函数的原型对象,并初始化构造函数,如果值则返回值。咱们也会在后续的原型(后续文章会对其介绍)中介绍,new 是隐式原型继承,Object.create 是显式原型继承
在这里,咱们按实现 new 的形式来解读 return new F()
。new F
后的实例的 __proto__
指向的是 F.prototype,而这个值曾经在第二步时指给了传来的 proto,所以就有了 new F().__proto__ = proto
或者你还是不太分明第三步,咱们联合例子,就高深莫测了
var obj = Object.create({name: 'johan'})
第三步的图解就成了这样:
这样就成了,obj 继承自 {name: johan}
这个对象,至于 F.prototype = {name: 'johan'}
,在调用完 Object.create
之后,也因为没人应用 F 函数而被引擎当作垃圾回收了,遂成了 obj.__proto__ = {name: 'johan'}
如此「原型式继承」就被传承下来了
其原理就是如此,简略来说,就是创立空(结构)函数,关联它的原型(实现继承)
Object.create(null)
在浏览源码时,常会看到 Object.create(null)
,用此初始化一个新对象,至于为什么用这个办法而不必 new Object 或者 {},是因为无论 new 还是字面量,都是继承自 Object 构造函数,而应用 Object.create(null),能失去一个没有任何继承痕迹的对象
var obj = Object.create(null)
不信,你能够打印 obj 试试
讲完了 new 和 Object.create,咱们来看看原型(后续文章会介绍)
系列文章
- 深刻了解 JavaScript- 开篇
- 深刻了解 JavaScript-JavaScript 是什么
- 深刻了解 JavaScript-JavaScript 由什么组成
- 深刻了解 JavaScript- 所有皆对象
- 深刻了解 JavaScript-Object(对象)
- 深刻了解 JavaScript-new 做了什么