JavaScript 原型与原型链详情


Posted in Javascript onNovember 02, 2021

前言:

JavaScript常被描述为一种「基于原型的语言」——每个对象都拥有一个「原型对象」,对象以其原型为模板、从原型继承属性和放法。原型对象也可能拥有原型,并从中继承属性和方法,一层一层以此类推。这种关系常被称为「原型链」,它解释了为何一个对象会拥有定义在其他对象中的属性和方法。

准确的说,这些属性和方法定义在Object的构造函数的prototype属性上,而非对象实例本身。

四句话道破原型与原型链:

  • 每个函数(类)天生自带一个属性prototype,属性值是一个对象,里面存储了当前类供实例使用的属性和方法 「(显示原型)」
  • 在浏览器默认给原型开辟的堆内存中有一个constructor属性:存储的是当前类本身(⚠️注意:自己开辟的堆内存中默认没有constructor属性,需要自己手动添加)「(构造函数)」
  • 每个对象都有一个__proto__属性,这个属性指向当前实例所属类的原型(不确定所属类,都指向Object.prototype)「(隐式原型)」
  • 当你试图获取一个对象的某个属性时,如果这个对象本身没有这个属性,那么它会去它的隐式原型__proto__(也就是它的构造函数的显示原型prototype)中查找。「(原型链)」

构造函数,原型与实例的关系:

每个构造函数(constructor)都有一个原型对象(prototype),原型对象(prototype)都包含一个指向构造函数(constructor)的指针,而实例(instance)都包含一个指向原型对象(__proto__)的内部指针

1、prototype(显式原型)

每个函数都有一个prototype属性

// 构造函数(类)
function Person(name){
    this.name = name
}
// new了一个实例 (对象)
var person = new Person('南玖')
console.log(person) //Person { name: '南玖' }
console.log(Person.prototype)  //构造函数(类)的原型 ----->对象
Person.prototype.age = 18  // 构造函数原型
console.log(person.age)  // 18

上面我们把这个函数Person的原型打印出来了,它指向的是一个对象,并且这个对象正是调用该构造函数而创建的实例的原型

JavaScript 原型与原型链详情

上面这张图表示的是构造函数与实例原型之间的关系,所以我们知道了构造函数的prototype属性指向的是一个对象。

那实例与实例原型之间的关系又是怎样的呢?这里就要提到__proto__属性了

2、__proto__(隐式原型)

从上面四句话中我们可以知道这是每一个Javascript对象(除null)都具有的一个属性,这个属性会指向该对象的原型(也就是实例原型)

因为在JavaScript中没有类的概念,为了实现类似继承的方式,通过__proto__将对象和原型联系起来组成原型链,的以让对象访问到不属于自己的属性。

那么我们就能够证明实例与实例原型之间的关系

console.log(person.__proto__)  //实例(对象)的原型--->对象

console.log(person.__proto__ === Person.prototype)  //实例的原型与构造函数的原型相等

JavaScript 原型与原型链详情

从上图我们可以看出实例对象与构造函数都可以指向原型,那么原型能不能指向构造函数或者是实例呢?

3、constructor(构造函数)

原型是没有属性指向实例的,因为一个构造函数可以创建多个实例对象;

从前面的四句话中我们知道「在浏览器默认给原型开辟的堆内存中有一个constructor属性」,所以原型也是可以指向构造函数的,这个属性就是「constructor

于是我们可以证明一下观点:

console.log(Person.prototype.constructor) //实例的显式原型的构造函数ƒ Person(name){this.name = name}
console.log(person.__proto__.constructor)  //实例的隐式原型的构造函数 ƒ Person(name){this.name = name}
console.log(person.__proto__.constructor === Person.prototype.constructor)//true 实例原型的构造函数与类的构造函数相等
console.log(Person === Person.prototype.constructor)  //true

JavaScript 原型与原型链详情

实例对象的__proto__是如何产生的?
我们知道当我们使用new 操作符时,生成的实例对象就拥有了__proto__属性

function Foo() {}
// 这个函数时Function的实例对象
// function是一个语法糖
// 内部其实调用了new Function()

所以可以说,在new的过程中,新对象被添加了__proto__属性并且链接到了构造函数的原型上。

4、new的原理

说简单点可以分为以下四步:

  • 新建一个空对象
  • 链接原型
  • 绑定this,执行构造函数
  • 返回新对象
function myNew() {
// 1.新建一个空对象
let obj = {}
// 2.获得构造函数
let con = arguments.__proto__.constructor
// 3.链接原型
obj.__proto__ = con.prototype
// 4.绑定this,执行构造函数
let res = con.apply(obj, arguments)
// 5.返回新对象
return typeof res === 'object' ? res : obj
}

5、原型链

说完了原型,我们再来看看什么是原型链?先来看一张图:

JavaScript 原型与原型链详情

这张图中,由__proto__串起来的链式关系,我们就称它为原型链

5.1 原型链的作用

原型链决定了JavaScript中继承的实现方式,当我们访问一个属性时,它的查找机制如下:

  • 访问对象实例属性,有的话直接返回,没有则通过__proto__去它的原型对象上查找
  • 原型对象上能找到的话则返回,找不到继续通过原型对象的__proto__查找
  • 一直往下找,直到找到Object.prototype,如果能找到则返回,找不到就返回undefined,不会再往下找了,因为Object.prototype.__proto__是null,说明了Object是所有对象的原型链顶层了。

从图中我们可以发现,所有对象都可以通过原型链最终找到 Object.prototype ,虽然 Object.prototype 也是一个对象,但是这个对象却不是 Object 创造的,而是引擎自己创建了 Object.prototype 。所以可以这样说,所有实例都是对象,但是对象不一定都是实例。

5.2 构造函数的__proto__是什么呢?

由上面的原型链的解释,我们应该能够理解构造函数的__proto__的,在JavaScript中所有东西都是对象,那么构造函数肯定也是对象,是对象就有__proto__

function Person(){}
console.log(Person.__proto__)
console.log(Function.prototype)
console.log(Person.__proto__===Function.prototype) // true

「这也说明了所有函数都是Function的实例」

那这么理解的话,Function.__proto__岂不是等于Function.prototype。。。。我们不妨来打印一下看看

Function.__proto__ === Function.prototype // true

打印出来确实是这样的。难道 Function.prototype 也是通过 new Function() 产生的吗?

答案是否定的,这个函数也是引擎自己创建的。首先引擎创建了 Object.prototype ,然后创建了 Function.prototype ,并且通过 __proto__ 将两者联系了起来。这里也很好的解释了上面的一个问题,为什么 let fun = Function.prototype.bind() 没有 prototype 属性。因为 Function.prototype 是引擎创建出来的对象,引擎认为不需要给这个对象添加 prototype 属性。

6、总结

  • Object 是所有对象的爸爸,所有对象都可以通过 __proto__ 找到它
  • Function 是所有函数的爸爸,所有函数都可以通过 __proto__ 找到它
  • Function.prototype Object.prototype 是两个特殊的对象,他们由引擎来创建
  • 除了以上两个特殊对象,其他对象都是通过构造器 new 出来的
  • 函数的 prototype 是一个对象,也就是原型
  • 对象的 __proto__ 指向原型, __proto__ 将对象和原型连接起来组成了原型链

到此这篇关于JavaScript 原型与原型链详情的文章就介绍到这了,更多相关JavaScript 原型与原型链内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
jQuery 加上最后自己的验证
Nov 04 Javascript
JavaScript与Image加载事件(onload)、加载状态(complete)
Feb 14 Javascript
bootstrap下拉列表与输入框组结合的样式调整
Oct 08 Javascript
js实现tab选项卡切换功能
Jan 13 Javascript
Javascript前端经典的面试题及答案
Mar 14 Javascript
JS库particles.js创建超炫背景粒子插件(附源码下载)
Sep 13 Javascript
JS实现点击发送验证码 xx秒后重新发送功能
Jul 30 Javascript
js通过循环多张图片实现动画效果
Dec 19 Javascript
easyUI使用分页过滤器对数据进行分页操作实例分析
Jun 01 Javascript
解决vue动态下拉菜单 有数据未反应的问题
Aug 06 Javascript
vue使用require.context实现动态注册路由
Dec 25 Vue.js
JavaScript原始值与包装对象的详细介绍
May 11 Javascript
javascript实现计算器功能详解流程
JS创建或填充任意长度数组的小技巧汇总
Oct 24 #Javascript
一文彻底理解js原生语法prototype,__proto__和constructor
Oct 24 #Javascript
javascript遍历对象的五种方式实例代码
Oct 24 #Javascript
低门槛开发iOS、Android、小程序应用的前端框架详解
Oct 16 #Javascript
基于angular实现树形二级表格
ajax请求前端跨域问题原因及解决方案
You might like
基于session_unset与session_destroy的区别详解
2013/06/03 PHP
Yii入门教程之Yii安装及hello world
2014/11/25 PHP
PHP实现合并两个排序链表的方法
2018/01/19 PHP
Ext javascript建立超链接,进行事件处理的实现方法
2009/03/22 Javascript
Whatever:hover 无需javascript让IE支持丰富伪类
2010/06/29 Javascript
简单的前端js+ajax 购物车框架(入门篇)
2011/10/29 Javascript
jQuery 取值、赋值的基本方法整理
2014/03/31 Javascript
Javascript编写俄罗斯方块思路及实例
2015/07/07 Javascript
浅谈Jquery中Ajax异步请求中的async参数的作用
2016/06/06 Javascript
nodejs简单实现操作arduino
2016/09/25 NodeJs
angular 基于ng-messages的表单验证实例
2017/05/04 Javascript
JS中LocalStorage与SessionStorage五种循序渐进的使用方法
2017/07/12 Javascript
JavaScript模拟文件拖选框样式v1.0的实例
2017/08/04 Javascript
30分钟用Node.js构建一个API服务器的步骤详解
2019/05/24 Javascript
微信小程序防止多次点击跳转和防止表单组件输入内容多次验证功能(函数防抖)
2019/09/19 Javascript
解决angular 使用原生拖拽页面卡顿及表单控件输入延迟问题
2020/04/21 Javascript
原生js+canvas实现下雪效果
2020/08/02 Javascript
Python导出数据到Excel可读取的CSV文件的方法
2015/05/12 Python
python连接数据库的方法
2017/10/19 Python
Python用imghdr模块识别图片格式实例解析
2018/01/11 Python
python版百度语音识别功能
2019/07/09 Python
彻底搞懂 python 中文乱码问题(深入分析)
2020/02/28 Python
Python自省及反射原理实例详解
2020/07/06 Python
CSS3 border-radius圆角的实现方法及用法详解
2020/09/14 HTML / CSS
HTML5中的postMessage API基本使用教程
2016/05/20 HTML / CSS
HTML5 Canvas渐进填充与透明实现图像的Mask效果
2013/07/11 HTML / CSS
巴黎卡诗加拿大官网:Kérastase加拿大
2018/11/12 全球购物
Boom手表官网:瑞典手表品牌,设计你的手表
2019/03/11 全球购物
临床医学专业毕业生的自我评价
2013/10/17 职场文书
《放飞蜻蜓》教学反思
2014/04/27 职场文书
祖国在我心中的演讲稿
2014/05/04 职场文书
购房协议书范本
2014/10/02 职场文书
保密工作整改情况汇报
2014/11/06 职场文书
乡镇党建工作总结2015
2015/05/19 职场文书
python实现大文本文件分割成多个小文件
2021/04/20 Python
Linux中一对多配置日志服务器的详细步骤
2022/07/23 Servers