Javascript中的原型链、prototype、__proto__的关系

2024-05-05 16:38

本文主要是介绍Javascript中的原型链、prototype、__proto__的关系,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文链接:http://shuaihua.cc/article/javascript/prototype-chain.php

Javascript中的原型链、prototype、__proto__的关系

上图是本宝宝用Illustrator制作的可视化信息图,希望能帮你理清Javascript对象与__proto__、prototype和原型链之间的关系。如果暂时看不懂也没关系,这篇文章让你从0变成1。

0、感性认识JS里的“德罗斯特效应”之原型链

如果你打开浏览器的控制面板,随便输入一个JS内置的构造器函数,比如Array,控制台输出的是一个名为Array的函数体,这好像并没有什么稀奇的,但是,当你接着输入Array.prototype,控制面板输出了一堆我们经常用到的Array构造器的方法,把目光转移到最下方,有一个叫__proto__的属性,好奇的点开,咦~?列表列出的不是Object构造器的方法么,里边有我们非常熟悉的hasOwnProperty还有toString等方法。我们常说的一切皆对象,一切对象皆null这么玄妙的编程界哲理和这种嵌套会不会有什么关系呢?更神奇的在下边。

当我继续深入,如果Array是构造器,那么控制面板输出的Array.prototype的所有属性中constructor又是什么构造器?点开看看,之后就像身处德罗斯特效应中一样,__proto__和constructor,还有Array构造器中常用的方法名不断的出现,一层套一层,一层层展开,没有尽头。。。

一、满满一堂证明题课

当你也发现了JS里这种循环往复、不断嵌套的规律后,仍然不相信自己的眼睛,于是决定亲自证明一下,哦不,是好几下。(对于初学者只靠空想太烧脑,于是本宝宝做了几个GIF动图。)

1、怎么证明你就是你?

这个问题可难不倒JS中的构造器们,一个证明秒秒钟证明“我就是我”,是颜色不一样的烟火。

拿Array举例,Array.prototype中有一个constructor属性,这个属性的值就是Array构造器自己!

Array.prototype.constructor === Array

2、“遗传进化链__proto__”,怎么证明一切皆对象?

所有的JS内置构造器都本是对象,这要long long ago说起。可是从什么线索开始向过去前进呢,原型链(我给他起了一个名字叫遗传进化链)就是突破口。所有JS构造器(当然不止构造器有)都有一个__proto__属性,这是原型链指针,指向遗传进化成它的“那个”。它“遗传”了“那个”的特性,而有进化出了它自有的特性。

JS内置构造器本是函数

从上图中发现,JS内置构造器其中之一的Array原本就是一个函数,而这个函数就是Function的prototype,所以Function.prototype有的方法,JS内置构造器都有,比如call()、apply()、bind()等(其实我们自定义的函数也是继承自Function.prototype,所以我们自己也可以定义构造器,创造属于自己的小小王国)。

而Function.prototype的进化链指针又指向了Object.prototype。

3、怎么证明到头来一切都是空?

不管你从那个属性开始,连续引用__proto__的指针,最后输出的那个值就是null。

空不异色,色不异空,空即是色,色即是空。

4、怎么证明所有JS内置构造器和自定义函数都是Function构造器的原型(prototype)。

不断强化这一认知,实践出真知。

①、String构造器的进化链指针__proto__指向Function构造器的原型

//String构造器
String.__Proto__ === Function.prototype

String构造器的进化链指针指向Function构造器的原型。

②、Number构造器的进化链指针__proto__指向Function构造器的原型

//Number构造器
Number.__Proto__ === Function.prototype

Number构造器的进化链指针指向Function构造器的原型。

③、Boolean构造器的进化链指针__proto__指向Function构造器的原型

//Boolean构造器
Boolean.__Proto__ === Function.prototype

Boolean构造器的进化链指针指向Function构造器的原型。

④、Array构造器的进化链指针__proto__指向Function构造器的原型

//Array构造器
Array.__Proto__ === Function.prototype

Array构造器的进化链指针指向Function构造器的原型。

⑤、没错,Function构造器的进化链指针__proto__也指向自己的原型

//Function构造器
Function.__Proto__ === Function.prototype

Function构造器的进化链指针指向Function构造器的原型。

⑥、Date构造器的进化链指针__proto__指向Function构造器的原型

//Date构造器
Date.__Proto__ === Function.prototype

Date构造器的进化链指针指向Function构造器的原型。

⑦;、Error构造器的进化链指针__proto__指向Function构造器的原型

//Error构造器
Error.__Proto__ === Function.prototype

Error构造器的进化链指针指向Function构造器的原型。

⑧、Object构造器的进化链指针__proto__指向Function构造器的原型

//Object构造器
Object.__Proto__ === Function.prototype

Object构造器的进化链指针指向Function构造器的原型。

⑨、RegExp构造器的进化链指针__proto__指向Function构造器的原型

//RegExp构造器
RegExp.__Proto__ === Function.prototype

RegExp构造器的进化链指针指向Function构造器的原型。

⑩、Event构造器的进化链指针__proto__指向Function构造器的原型

//Event构造器
Event.__Proto__ === Function.prototype

Event构造器的进化链指针指向Function构造器的原型。

这里需要注意所有构造器的prototype都是对象(object)类型,只有Function.prototype是函数(function)类型,这是为了保证函数构造器们的__proto__指向的都是函数。

二,咦?JSON和Math哪去啦?

JS内置的构造器函数都可以使用new关键字实例化一个对象,我们称实例化后的这个对象就是某某构造器的一个实例。就像我们每一个“人”都是“人类”这个构造器函数的一个实例

//实例化一个String构造函数
var str = new String("Hi, today! ");

实例化一个String构造函数str

既然上边10个构造器函数都能这样实例化对象,那么JSON和Math是不是也可以用new 关键字实例化呢?试试看!

//尝试实例化JSON和Math两个构造器函数
var json = new JSON();
var Math = new Math();

JSON和Math不是构造器函数,他们是普通的对象

哦No,不可以。JSON和Math不是构造器函数,他们是普通的对象。

上边提到过,只有构造器函数才能使用new 关键字实例化一个对象,而JSON和Math已经是对象了,所以我们可以不用实例化直接使用JSON和Math中的属性和方法~~(我们实例化的目就是想用实例化后的对象里的属性和方法,那么既然JSON和Math已经是对象了,就省去实例化的操作喽。当然,能实例化有能实例化的好处~)

所以JSON和Math不属于10个构造器函数,但他们12个共同属于Javascript的内置对象。

三、__proto__进化链指针设计为什么如此重要!!

javascript中为什么会有__proto__原型链的设计,不放做一个小实验先。

//实例化一个String对象
var str = new String("Hi!")

我们先实例化一个String对象并将其赋值给str这个变量,然后我们输出这个str

输出String实例化后的对象

从str输出的内容来看,str有四个属性,分别是0、1、2、length

//我们一个个输出这些属性
console.log(str[0]) //H
console.log(str[1]) //i
console.log(str[2]) //!
console.log(str['length']) //3

这毋庸置疑,但是接着往下看。

//charAt()是str对象中不存在的属性方法,但是没有报错,依然可以输出!
str.charAt(0) // 输出'H'

charAt()是str对象中不存在的属性方法,但是没有报错,依然可以输出!

调用str中不存在的属性方法

这是为什么?这就是进化链__proto__的用处。

str这个对象本身的确没有charAt()这个方法,但是str的进化链上存在这个属性方法,那么charAt()这个方法在进化链的那个节点上呢?

String.prototype拥有charAt这个方法,而str的__proto__指针指向String.prototype

哦~,原来String.prototype拥有charAt这个方法,而str的__proto__指针指向String.prototype

这么说的话,str.__proto__.__proto__指向的对象所拥有的属性str也都可以直接用喽?答案是肯定的!

str.__proto__.__proto__指向的对象所拥有的属性中有一个hasOwnProperty属性

看到str.__proto__.__proto__指向的对象所拥有的属性中有一个hasOwnProperty属性方法了么,str可以直接使用这个属性方法。

在验证之前先说下str.__proto__.__proto__指向了谁?指向的是Object的prototype属性。

str.__proto__.__proto__ === Object.prototype // true

Object.prototype.hasOwnProperty()属性方法用来检验一个对象是否自己拥有一个属性而非通过进化链__proto__继承来的属性。

好的,实验开始:

//检查str是否拥有length属性
str.hasOwnProperty('length') //true
//检查str是否拥有0属性(str.charAt(0)的输出是'H')
str.hasOwnProperty(0) //true
//检查str是否拥有1属性
str.hasOwnProperty(1) //true
//检查str是否拥有2属性
str.hasOwnProperty(2) //true
//检查str是否拥有3属性(str的length是3,所以不存在索引3这个属性)
str.hasOwnProperty(3) //false
//str是否拥有hasOwnProperty这个属性呢?答案是否定的。
str.hasOwnProperty('hasOwnProperty') //false

检验str的自有属性

四、结语

现在再来看这张图,是不是思路清晰多了呢!

Javascript中的原型链、prototype、__proto__的关系

当你弄清楚了原型链(我喜欢叫他进化链)__proto__,prototype之间的关系,还有Javascript中12个内置对象,其中10个函数类型,2个对象类型。再来学习这12个内置对象的属性和属性方法是不是如鱼得水,心里跟明镜似的~

对于Javascrip初学者,一时半会肯定还是搞不清楚,唯一的办法就是多看、多想、多练、多总结、多分享~,学习的本质无非就是这些嘛,教育的本质无非就是教会你学习你所感兴趣的嘛。

原文链接:http://shuaihua.cc/article/javascript/prototype-chain.php

这篇关于Javascript中的原型链、prototype、__proto__的关系的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/962160

相关文章

SpringBoot集成XXL-JOB实现任务管理全流程

《SpringBoot集成XXL-JOB实现任务管理全流程》XXL-JOB是一款轻量级分布式任务调度平台,功能丰富、界面简洁、易于扩展,本文介绍如何通过SpringBoot项目,使用RestTempl... 目录一、前言二、项目结构简述三、Maven 依赖四、Controller 代码详解五、Service

Java中HashMap的用法详细介绍

《Java中HashMap的用法详细介绍》JavaHashMap是一种高效的数据结构,用于存储键值对,它是基于哈希表实现的,提供快速的插入、删除和查找操作,:本文主要介绍Java中HashMap... 目录一.HashMap1.基本概念2.底层数据结构:3.HashCode和equals方法为什么重写Has

Java 正则表达式的使用实战案例

《Java正则表达式的使用实战案例》本文详细介绍了Java正则表达式的使用方法,涵盖语法细节、核心类方法、高级特性及实战案例,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录一、正则表达式语法详解1. 基础字符匹配2. 字符类([]定义)3. 量词(控制匹配次数)4. 边

Java Scanner类解析与实战教程

《JavaScanner类解析与实战教程》JavaScanner类(java.util包)是文本输入解析工具,支持基本类型和字符串读取,基于Readable接口与正则分隔符实现,适用于控制台、文件输... 目录一、核心设计与工作原理1.底层依赖2.解析机制A.核心逻辑基于分隔符(delimiter)和模式匹

Java中的stream流分组示例详解

《Java中的stream流分组示例详解》Java8StreamAPI以函数式风格处理集合数据,支持分组、统计等操作,可按单/多字段分组,使用String、Map.Entry或Java16record... 目录什么是stream流1、根据某个字段分组2、按多个字段分组(组合分组)1、方法一:使用 Stri

Java+AI驱动实现PDF文件数据提取与解析

《Java+AI驱动实现PDF文件数据提取与解析》本文将和大家分享一套基于AI的体检报告智能评估方案,详细介绍从PDF上传、内容提取到AI分析、数据存储的全流程自动化实现方法,感兴趣的可以了解下... 目录一、核心流程:从上传到评估的完整链路二、第一步:解析 PDF,提取体检报告内容1. 引入依赖2. 封装

使用Spring Cache本地缓存示例代码

《使用SpringCache本地缓存示例代码》缓存是提高应用程序性能的重要手段,通过将频繁访问的数据存储在内存中,可以减少数据库访问次数,从而加速数据读取,:本文主要介绍使用SpringCac... 目录一、Spring Cache简介核心特点:二、基础配置1. 添加依赖2. 启用缓存3. 缓存配置方案方案

Java实现复杂查询优化的7个技巧小结

《Java实现复杂查询优化的7个技巧小结》在Java项目中,复杂查询是开发者面临的“硬骨头”,本文将通过7个实战技巧,结合代码示例和性能对比,手把手教你如何让复杂查询变得优雅,大家可以根据需求进行选择... 目录一、复杂查询的痛点:为何你的代码“又臭又长”1.1冗余变量与中间状态1.2重复查询与性能陷阱1.

深度剖析SpringBoot日志性能提升的原因与解决

《深度剖析SpringBoot日志性能提升的原因与解决》日志记录本该是辅助工具,却为何成了性能瓶颈,SpringBoot如何用代码彻底破解日志导致的高延迟问题,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言第一章:日志性能陷阱的底层原理1.1 日志级别的“双刃剑”效应1.2 同步日志的“吞吐量杀手”

Spring创建Bean的八种主要方式详解

《Spring创建Bean的八种主要方式详解》Spring(尤其是SpringBoot)提供了多种方式来让容器创建和管理Bean,@Component、@Configuration+@Bean、@En... 目录引言一、Spring 创建 Bean 的 8 种主要方式1. @Component 及其衍生注解