文章目录

  • 1. 前言
  • 2. _ _ proto _ _ 属性
  • 3. prototype属性
  • 4. constructor属性
  • 5. 总结

提示:不要排斥,静下心来,认真读完,你就搞懂了!(可以先看一下最后的总结部分再回过头来完整看完)

1. 前言

  作为一名前端工程师,必须搞懂JS中的prototype__proto__constructor属性,相信很多初学者对这些属性存在许多困惑,容易把它们混淆,本文旨在帮助大家理清它们之间的关系并彻底搞懂它们。这里说明一点,__proto__属性的两边是各由两个下划线构成(这里为了方便大家看清,在两下划线之间加入了一个空格:_ _proto_ _,读作“dunder proto”,“double underscore proto”的缩写),实际上,该属性在ES标准定义中的名字应该是[[Prototype]],具体实现是由浏览器代理自己实现,谷歌浏览器的实现就是将[[Prototype]]命名为__proto__,大家清楚这个标准定义与具体实现的区别即可(名字有所差异,功能是一样的),可以通过该方式检测引擎是否支持这个属性:Object.getPrototypeOf({__proto__: null}) === null。本文基于谷歌浏览器(版本 72.0.3626.121)的实验结果所得。
   现在正式开始! 让我们从如下一个简单的例子展开讨论,并配以相关的图帮助理解:

function Foo() {...};
let f1 = new Foo();
  • 1
  • 2

以上代码表示创建一个构造函数Foo(),并用new关键字实例化该构造函数得到一个实例化对象f1。这里稍微补充一下new操作符将函数作为构造器进行调用时的过程:函数被调用,然后新创建一个对象,并且成了函数的上下文(也就是此时函数内部的this是指向该新创建的对象,这意味着我们可以在构造器函数内部通过this参数初始化值),最后返回该新对象的引用,详细请看:详解JavaScript中的new操作符。虽然是简简单单的两行代码,然而它们背后的关系却是错综复杂的,如下图所示:
看到这图别怕,让我们一步步剖析,彻底搞懂它们!
  图的说明:右下角为图例,红色箭头表示__proto__属性指向、绿色箭头表示prototype属性的指向、棕色实线箭头表示本身具有的constructor属性的指向,棕色虚线箭头表示继承而来的constructor属性的指向;蓝色方块表示对象,浅绿色方块表示函数(这里为了更好看清,Foo()仅代表是函数,并不是指执行函数Foo后得到的结果,图中的其他函数同理)。图的中间部分即为它们之间的联系,图的最左边即为例子代码。

2. _ _ proto _ _ 属性

  首先,我们需要牢记两点:①__proto__constructor属性是对象所独有的;② prototype属性是函数所独有的。但是由于JS中函数也是一种对象,所以函数也拥有__proto__constructor属性,这点是致使我们产生困惑的很大原因之一。上图有点复杂,我们把它按照属性分别拆开,然后进行分析:

  第一,这里我们仅留下 __proto__ 属性,它是对象所独有的,可以看到__proto__属性都是由一个对象指向一个对象,即指向它们的原型对象(也可以理解为父对象),那么这个属性的作用是什么呢?它的作用就是当访问一个对象的属性时,如果该对象内部不存在这个属性,那么就会去它的__proto__属性所指向的那个对象(可以理解为父对象)里找,如果父对象也不存在这个属性,则继续往父对象的__proto__属性所指向的那个对象(可以理解为爷爷对象)里找,如果还没找到,则继续往上找…直到原型链顶端null(可以理解为原始人。。。),再往上找就相当于在null上取值,会报错(可以理解为,再往上就已经不是“人”的范畴了,找不到了,到此结束,null为原型链的终点),由以上这种通过__proto__属性来连接对象直到null的一条链即为我们所谓的原型链
  其实我们平时调用的字符串方法、数组方法、对象方法、函数方法等都是靠__proto__继承而来的。

3. prototype属性

  第二,接下来我们看 prototype 属性:
  prototype属性,别忘了一点,就是我们前面提到要牢记的两点中的第二点,它是函数所独有的,它是从一个函数指向一个对象。它的含义是函数的原型对象,也就是这个函数(其实所有函数都可以作为构造函数)所创建的实例的原型对象,由此可知:f1.__proto__ === Foo.prototype,它们两个完全一样。那prototype属性的作用又是什么呢?它的作用就是包含可以由特定类型的所有实例共享的属性和方法,也就是让该函数所实例化的对象们都可以找到公用的属性和方法。任何函数在创建的时候,其实会默认同时创建该函数的prototype对象。

4. constructor属性

  最后,我们来看一下 constructor 属性:
  constructor属性也是对象才拥有的,它是从一个对象指向一个函数,含义就是指向该对象的构造函数,每个对象都有构造函数(本身拥有或继承而来,继承而来的要结合__proto__属性查看会更清楚点,如下图所示),从上图中可以看出Function这个对象比较特殊,它的构造函数就是它自己(因为Function可以看成是一个函数,也可以是一个对象),所有函数和对象最终都是由Function构造函数得来,所以constructor属性的终点就是Function这个函数。

  感谢网友的指出,这里解释一下上段中“每个对象都有构造函数”这句话。这里的意思是每个对象都可以找到其对应的constructor,因为创建对象的前提是需要有constructor,而这个constructor可能是对象自己本身显式定义的或者通过__proto__在原型链中找到的。而单从constructor这个属性来讲,只有prototype对象才有。每个函数在创建的时候,JS会同时创建一个该函数对应的prototype对象,而函数创建的对象.__proto__ === 该函数.prototype,该函数.prototype.constructor===该函数本身,故通过函数创建的对象即使自己没有constructor属性,它也能通过__proto__找到对应的constructor,所以任何对象最终都可以找到其构造函数(null如果当成对象的话,将null除外)。如下:

5. 总结

   总结一下:

  1. 我们需要牢记两点:①__proto__constructor属性是对象所独有的;② prototype属性是函数所独有的,因为函数也是一种对象,所以函数也拥有__proto__constructor属性。
  2. __proto__属性的作用就是当访问一个对象的属性时,如果该对象内部不存在这个属性,那么就会去它的__proto__属性所指向的那个对象(父对象)里找,一直找,直到__proto__属性的终点null,再往上找就相当于在null上取值,会报错。通过__proto__属性将对象连接起来的这条链路即我们所谓的原型链
  3. prototype属性的作用就是让该函数所实例化的对象们都可以找到公用的属性和方法,即f1.__proto__ === Foo.prototype
  4. constructor属性的含义就是指向该对象的构造函数,所有函数(此时看成对象了)最终的构造函数都指向Function

  本文就此结束了,希望对那些对JS中的prototype__proto__constructor属性有困惑的同学有所帮助。

最后,感谢这两篇博文,本文中的部分内容参考自这两篇博文:
https://www.cnblogs.com/xiaohuochai/p/5721552.html
https://www.cnblogs.com/Narcotic/p/6899088.html


http://www.taodudu.cc/news/show-2887854.html

相关文章:

  • SpringBoot+Vue博客项目中遇到的坑
  • python 条件概率_NLTK中的条件概率分布
  • 帮你彻底搞懂JS中的prototype、__proto__与constructor(图解)
  • 旧金山犯罪预测与可视化分析
  • 2022.10.28 英语背诵
  • IT行业人才招聘回顾与展望
  • 《机器学习实战》学习笔记(四):基于概率论的分类方法 - 朴素贝叶斯
  • 转贴:ubuntu 7.10 常用软件与编程环境搭建
  • 简化控制器内容
  • 东邪、西毒、南帝、北丐、中神通五大操作系统之华山论剑(独评)
  • kafka 0.10.0 producer java代码实现
  • beaglebone black下接nrf24l01与RFID标签的通信(基于EZSDK linux平台)
  • 阿萨德撒的撒的撒的
  • Android以太坊钱包全部功能-基于web3j实现
  • ARM Linux 内核 panic 之cache 一致性 ——cci-400 cache一致互联
  • Selenium 爬虫应用的学习
  • am335x linux内核烧写_am335x文件系统烧写问题
  • 小爱同学指令大全_小爱同学有哪些隐藏功能?小爱同学实用隐藏功能大汇总
  • 小天才z6官方禁用怎么关闭_我告诉你小天才z6隐藏功能
  • 你不能不知道的荣耀V40隐藏功能
  • 科学计算机隐藏功能,经常用手机计算器的抓紧看看,原来还隐藏着3个功能,涨知识了...
  • 原来华为手机的拨号键盘除了打电话,还有这些隐藏功能,涨知识了
  • vivo计算机的隐藏功能介绍,六大vivo隐藏黑科技功能 绝对有你不知道的
  • 华为android10手机隐藏小游戏,华为手机10个实用好玩的隐藏功能
  • 别说华为语音助手不智能了,这3大隐藏功能都知道吗?实用又贴心
  • vue实现select下拉显示隐藏功能【详细功能拓展】
  • MFC对话框部分区域的显示和隐藏功能的实现
  • 手机计算机藏应用,手机“计算器”隐藏功能,一键把隐私照片加密
  • 三星note20u计算机功能,三星Note20Ultra隐藏功能有哪些-有哪些使用技巧
  • 酷开系统这些隐藏功能你用过几个?

这一篇彻底搞懂JS中的prototype、__proto__与constructor真的很好相关推荐

  1. 帮你彻底搞懂JS中的prototype、__proto__与constructor(图解)

    帮你彻底搞懂JS中的prototype.__proto__与constructor(图解) 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明. 本文 ...

  2. (转)帮你彻底搞懂JS中的prototype、__proto__与constructor(图解)

    文章目录 1. 前言 2. _ _ proto _ _ 属性 3. prototype属性 4. constructor属性 5. 总结 提示:不要排斥,静下心来,认真读完,你就搞懂了!(可以先看一下 ...

  3. 彻底搞懂 JS 中 this 机制

    彻底搞懂 JS 中 this 机制 摘要:本文属于原创,欢迎转载,转载请保留出处:https://github.com/jasonGeng88/blog 目录 this 是什么 this 的四种绑定规 ...

  4. 彻底搞懂 JS 中 this 机制 1

    彻底搞懂 JS 中 this 机制 摘要:本文属于原创,欢迎转载,转载请保留出处:https://github.com/jasonGeng88/blog 目录 this 是什么 this 的四种绑定规 ...

  5. 一文搞懂JS中的赋值·浅拷贝·深拷贝

    前言 为什么写拷贝这篇文章?同事有一天提到了拷贝,他说赋值就是一种浅拷贝方式,另一个同事说赋值和浅拷贝并不相同.我也有些疑惑,于是我去MDN搜一下拷贝相关内容,发现并没有关于拷贝的实质概念,没有办法只 ...

  6. JS中的prototype、__proto__与constructor(图解)

    作为一名前端工程师,必须搞懂JS中的prototype.__proto__与constructor属性,相信很多初学者对这些属性存在许多困惑,容易把它们混淆,本文旨在帮助大家理清它们之间的关系并彻底搞 ...

  7. JS中的prototype、__proto__与constructor

    作为一名前端工程师,必须搞懂JS中的prototype.__proto__与constructor属性,相信很多初学者对这些属性存在许多困惑,容易把它们混淆,本文旨在帮助大家理清它们之间的关系并彻底搞 ...

  8. java 自旋锁_搞懂Java中的自旋锁

    轻松搞懂Java中的自旋锁 前言 在之前的文章<一文彻底搞懂面试中常问的各种"锁">中介绍了Java中的各种"锁",可能对于不是很了解这些概念的同学 ...

  9. 一文搞懂Qt中的颜色渐变(QGradient Class)

    一文搞懂Qt中的颜色渐变(QGradient Class) 1, 快速开始! Qt中与颜色渐变有关的类是QGradient 其中它又有三个子类:QLinearGradient.QRadialGradi ...

最新文章

  1. linux oracle dblink 访问 postgresql_从Oracle到PG-PostgreSQL数据库参数配置和查看
  2. Linux学习笔记重新梳理20180702 之 yum软件包管理器
  3. 【带着canvas去流浪(10)】文字烟花
  4. 台湾大学林轩田机器学习基石课程学习笔记12 -- Nonlinear Transformation
  5. 论文浅尝 - ICML2020 | 跨域对齐的图最优运输算法
  6. U盘容量显示错误修正
  7. python的matplotlib库怎么安装_为Python安装matplotlib库
  8. 分布科技荣登海南省实施区块链应用示范揭榜工程名单
  9. 添加服务oracle,oracle 11g(四)给oracle添加为系统服务(脚本)
  10. 【转】字符串和浮点数格式化输出小结
  11. 【Thread】java类Thread中提供了检测线程是否中断的方法,说一说你的了解?
  12. 移远EC20基站定位
  13. Java Web 项目基于IDEA的增量补丁打包插件
  14. 将PC端固定布局页面改成移动端流体布局。
  15. centos7 C++ 使用libjpeg-turbo (让jpg 转bmp以及bmp转jpg)
  16. luogu P2184 贪婪大陆
  17. 万恶的less-loader
  18. windows server 2008r2 更新失败解决方案
  19. Thoth多因子策略
  20. 有没有u盘数据恢复软件免费版?u盘数据恢复软件哪个好?

热门文章

  1. EOJ 3265 七巧板
  2. 【Fracturing amp; Destruction】Unity3D的物体爆裂、炸裂、碎裂效果
  3. Linux:Nginx 正向代理实现内网访问互联网
  4. TMS320C6678开发笔记---IBL编译与分析4
  5. Jason Brown的R快速入门方法
  6. Leaflet 和 Cesium 加载纠偏后高德地图在线瓦片,高德地图最新最全在线瓦片地址
  7. rip neighbor_借助众筹平台Neighbor.ly透明地提高公民素质
  8. MATLAB 车牌识别程序介绍 SVM、神经网络[毕业设计]
  9. 家用 NAS 服务器搭建 | 前篇
  10. io输出pwm且占空比和频率同时可调驱动实现