深入理解JavaScript-Object.create

简介: 深入理解JavaScript-Object.create

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;

image.png


第二步:F.prototype = proto

即将 F.prototype 赋值为传入的 proto,如此就打破了F.prototype = F.prototype 以及 F.prototype.constructor = F ,它们的关系为

image.png


第三步:return new F()

第三步的解读有点费解,因为这里涉及到 new 的操作,在 new 改变了对象 中我们说过,new 会创建一个对象,并将这个对象的隐式原型(__proto__) 指向构造函数的原型对象,并初始化构造函数,如果值则返回值。我们也会在后续的原型(后续文章会对其介绍) 中介绍,new 是隐式原型继承,Object.create 是显式原型继承


在这里,我们按实现 new 的方式来解读 return new F()new F 后的实例的 __proto__ 指向的是 F.prototype,而这个值已经在第二步时指给了传来的 proto,所以就有了new F().__proto__ = proto

image.png


或许你还是不太清楚第三步,我们结合例子,就一目了然了

var obj = Object.create({name: 'johan'})


第三步的图解就成了这样:

image.png


这样就成了, 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 前端开发 小程序
js 实现浏览器下载视频2种方法
js 实现浏览器下载视频2种方法
1604 0
|
存储 监控 算法
【Linux】根文件目录详情
【Linux】根文件目录详情
227 1
|
11月前
|
小程序 JavaScript
微信小程序之input组件及其获取用户输入信息
微信小程序之input组件及其获取用户输入信息
259 2
|
11月前
|
存储 分布式计算 NoSQL
大数据-136 - ClickHouse 集群 表引擎详解1 - 日志、Log、Memory、Merge
大数据-136 - ClickHouse 集群 表引擎详解1 - 日志、Log、Memory、Merge
275 0
|
程序员 开发者
IDEA插件-Rainbow Brackets彩虹括号
DEA插件-Rainbow Brackets是一款用于增强IDEA开发环境的工具,它可以帮助开发者更好地管理和展示代码中的括号。
3328 0
IDEA插件-Rainbow Brackets彩虹括号
|
前端开发
css动画(仿微信聊天页面)
css动画(仿微信聊天页面)
|
C++ 开发者
C++课程设计:图书管理系统【附源码】
图书管理系统,C++ 实现,附全套源码
5602 2
C++课程设计:图书管理系统【附源码】
|
应用服务中间件 nginx Windows
[emerg] 15060#200: bind() to 0.0.0.0:80 failed (10013: An attempt was made to access a socket ......
[emerg] 15060#200: bind() to 0.0.0.0:80 failed (10013: An attempt was made to access a socket ......
343 0
|
Web App开发 前端开发 Java
Netty-SocketIO:最好用的Java版即时消息推送
版权声明:欢迎转载,请注明沉默王二原创。 https://blog.csdn.net/qing_gee/article/details/52525677 Netty-SocketIO是一个开源的、基于Netty的、Java版的即时消息推送项目。
8205 1
|
监控 JavaScript 算法
电线电缆行业智能生产管理MES系统解决方案
电线电缆行业生产管理现状:电线电缆行业是典型的重资产行业,原材料成本高、产品价值高、资金占用大、产品规格型号多达数万种, BOM管理繁杂,现场管理粗放,订单经常会合并或拆分生产,对排程要求高。
553 0
电线电缆行业智能生产管理MES系统解决方案