详解JavaScript原型与原型链


Posted in Javascript onNovember 16, 2020

正如一些面向对象语言中所实现的那样,在JavaScript中我们有时也需要创建一个拥有公共函数与属性的类作为父类来减少代码重复、实现类型检查与实现更加清晰地代码结构。在JavaScript中,继承是通过原型链实现的。了解JavaScript的继承与原型链之前首先需要了解JavaScript中对象创建的方式。

在JavaScript中创建对象

JavaScript中对象创建的方式有两种:工厂方法(Factory Functions)、构造器方法(Constructor Functions) 。

工厂方法

工厂方法在编程领域是一个非类或构造器的返回对象的方法。在JavaScript中,任何返回不使用new关键词创建对象的方法都是工厂方法。

function person(firstName, lastName, age) {
 const person = {};
 person.firstName = firstName;
 person.lastName = lastName;
 person.age = age;
 return person;
}

构造器方法

构造器方法和工厂方法的区别仅在用例和命名规范上。命名规范上一个构造器方法的名字开头字母需要大写,我们需要通过new关键词来调用构造器方法生成实例。这个实例之后便可以通过instanceof关键词来检查。

function Person(firstName, lastName, age) {
 this.firstName = firstName;
 this.lastName = lastName;
 this.age = age;
}

new的行为

当同时在工厂方法和构造器方法上使用new关键词创建时,工厂方法创建出的对象的__proto__属性指向Object.prototype,构造器方法创建出的对象的__proto__属性指向本身的Xxx.prototype。

const mike = new person('mike', 'grand', 23);

mike.__proto__	// Object.prototype
const jack = new Person('jack', 'grand', 23);

jack.__proto__	// Person.protytype这里的prototype指向Person的Prototype Object
jack.__proto__.__proto__	// Object.prototype

new关键词在后台为构造器方法执行了以下几步

  1. 在构造器方法内创建一个新对象并将其赋值到this上
  2. 设置对象的[[Prototype]]和__proto__为原型的构造函数,这一步也让新对象的构造函数在构造新对象时被添加到原型链上
  3. 如果这个方法内没有返回object、function或array类型的结果,就返回this
  4. 如果这个方法内没有返回任何值则返回this

下面是一个展示new关键词在JavaScript引擎当中执行效果的伪代码,注释当中的是用来示范new关键词添加语句的伪代码

function Person(firstName, lastName, age) {
 // this = {};
 // this.__proto__ = Person.prototype;
 this.firstName = firstName;
 this.lastName = lastName;
 this.age = age;
 // return this;
}

在构造器方法上没有返回值所以后台创建的this将被返回,而工厂方法内由于返回了对象所以后台不再添加return this自然返回的内容将不一致。

如果没有在构造器方法前使用new关键词,而将构造器方法直接调用执行,其仅作为一个方法来被执行而非构造器。

const bob = Person('bob', 'grand', 23);
bob	// undefined. 因为这里Person当作方法直接调用了且没有返回值
window.firstName	// bob. 函数内的this将指向全局作用域,导致意外操作

继承与原型链

原型

原型(Prototype)可以认为是一个JavaScript方法的属性,每次在JavaScript代码中创建方法时,JavaScript引擎会将一个名为prototype的属性添加上去,这个prototype属性是一个对象(原型对象),这个对象默认有一个constructor属性指向原方法对象。任何添加到prototype的属性和方法都在这个对象里面,所有该类实例共享这个原型对象,实例对象的__proto__属性指向这个对象,方法的prototype属性指向这个对象。

在ECMAScript的标准里object.[[Prototype]]是访问原型的方法,但在ECMAScript 2015中用Object.getPrototypeOf()和Object.setPrototypeOf()来替代。等价的__proto__是多数浏览器使用的事实上的但是非标准的实现。

function Person(firstName, lastName, age) {
 this.firstName = firstName;
 this.lastName = lastName;
 this.age = age;
}

Person.prototype === Person.prototype.constructor.prototype	// 指向Person的原型对象
Person.prototype.constructor === Person	// 指向Person方法对象

let bob = new Person("Bob", "Ross", 21);
Person.prototype === bob.__proto__;	// true

let alex = new Person("Alex", "Wang", 21);
Person.prototype === alex.__proto__;	// true
alex.__proto__ === bob.__proto__;	// true

原型链

首先我们需要了解对象查找机制。当我们使用一个对象的属性时,JavaScript引擎会首先查找本对象里是否有对应属性,如果没有则去对象的原型里查找属性,如果没有则去对象的原型对象的原型对象里查找属性,直至查询到对象的__proto__为null的时候停止。

详解JavaScript原型与原型链

const obj = {};
console.log(obj);	// [object Object]	obj的toString()方法从Object的原型中查找到并使用

function Person(firstName, lastName, age) {
 this.firstName = firstName;
 this.lastName = lastName;
 this.age = age;
 // 在Person.prototype上定义了toString覆写了Object.prototype上的toString
 Person.prototype.toString = function() {
  return `${this.firstName} It Is`;
 }
}

let bob = new Person("Bob", "Ross", 21);
let alex = new Person("Alex", "Wang", 21);
console.log(bob);	// Bob It Is
console.log(alex);	// Alex It Is

以上就是详解JavaScript原型与原型链的详细内容,更多关于JavaScript原型与原型链的资料请关注三水点靠木其它相关文章!

Javascript 相关文章推荐
jquery的键盘事件修改代码
Feb 24 Javascript
js 判断checkbox是否选中的操作方法
Nov 09 Javascript
JS获取图片实际宽高及根据图片大小进行自适应
Aug 11 Javascript
javascript单引号和双引号的区别和处理
May 14 Javascript
firefox浏览器用jquery.uploadify插件上传时报HTTP 302错误
Mar 01 Javascript
[原创]Javascript 实现广告后加载 可加载百度谷歌联盟广告
May 11 Javascript
node.js学习之base64编码解码
Oct 21 Javascript
如何在AngularJs中调用第三方插件库
May 21 Javascript
详解Vue Elememt-UI构建管理后台
Feb 27 Javascript
JavaScript实现浅拷贝与深拷贝的方法分析
Jul 05 Javascript
OpenLayer3自定义测量控件MeasureTool
Sep 28 Javascript
Express 配置HTML页面访问的实现
Nov 01 Javascript
详解JavaScript执行模型
Nov 16 #Javascript
Vue 实现拨打电话操作
Nov 16 #Javascript
微信小程序实现页面左右滑动
Nov 16 #Javascript
Vertx基于EventBus发送接受自定义对象
Nov 16 #Javascript
vue+echarts+datav大屏数据展示及实现中国地图省市县下钻功能
Nov 16 #Javascript
angular8.5集成TinyMce5的使用和详细配置(推荐)
Nov 16 #Javascript
js实现纯前端压缩图片
Nov 16 #Javascript
You might like
关于我转生变成史莱姆这档事:第二季PV上线,萌王2021年回归
2020/05/06 日漫
支持汉转拼和拼音分词的PHP中文工具类ChineseUtil
2018/02/23 PHP
JS启动应用程序的一个简单例子
2008/05/11 Javascript
遍历jquery对象的代码分享
2011/11/02 Javascript
js 自制滚动条的小例子
2013/03/16 Javascript
在Node.js应用中使用Redis的方法简介
2015/06/24 Javascript
javascript下使用Promise封装FileReader
2016/02/19 Javascript
基于JavaScript实现的折半查找算法示例
2017/04/14 Javascript
老生常谈jacascript DOM节点获取
2017/04/17 Javascript
JavaScript表单验证实现代码
2017/05/22 Javascript
微信禁止下拉查看URL的处理方法
2017/09/28 Javascript
Vue使用vux-ui自定义表单验证遇到的问题及解决方法
2018/05/10 Javascript
微信小程序项目总结之点赞 删除列表 分享功能
2018/06/25 Javascript
理解Proxy及使用Proxy实现vue数据双向绑定操作
2020/07/18 Javascript
[01:22:29]真视界:2019年国际邀请赛总决赛
2020/01/29 DOTA
压缩包密码破解示例分享(类似典破解)
2014/01/17 Python
Python 字典dict使用介绍
2014/11/30 Python
python中dir函数用法分析
2015/04/17 Python
Python3搜索及替换文件中文本的方法
2015/05/22 Python
python编程实现随机生成多个椭圆实例代码
2018/01/03 Python
Python实用工具FuckIt.py介绍
2019/07/02 Python
Python及Pycharm安装方法图文教程
2019/08/05 Python
Python搭建代理IP池实现获取IP的方法
2019/10/27 Python
简述python Scrapy框架
2020/08/17 Python
基于python判断字符串括号是否闭合{}[]()
2020/09/21 Python
python 基于Apscheduler实现定时任务
2020/12/15 Python
HTML中使用SVG与SVG预定义形状元素介绍
2013/06/28 HTML / CSS
使用HTML5 Geolocation实现一个距离追踪器
2018/04/09 HTML / CSS
佳能德国网上商店:Canon德国
2017/03/18 全球购物
西铁城美国官方网站:Citizen Watch美国
2019/11/08 全球购物
任课老师推荐信范文
2013/11/24 职场文书
优秀共产党员先进事迹
2014/01/27 职场文书
详细的大学生创业计划书模板
2014/01/27 职场文书
2014年高考决心书
2014/03/11 职场文书
实习公司领导推荐函
2014/05/21 职场文书
Redis之RedisTemplate配置方式(序列和反序列化)
2022/03/13 Redis