前言

原型原型链应该是被大多数前端er说烂的词,但是应该还有很多人不能完整的解释这两个内容,当然也包括我自己。

最早一篇原型链文章写于2019年07月,那个时候也是费了老大劲才理解到了七八成,到现在基本上忘的差不多了。时隔两年,兴趣所向重新开始复盘一下原型原型链的内容。

JavaScript中的对象

JavaScript中,对象被称为是一系列属性的集合。

创建对象的方式也有很多种,最常见的一种就是双花括号的形式:

var obj = {};
obj.name = '小土豆';
obj.age = 18;

这种方式实际上是下面这种方式的语法糖

var obj = new Object();
obj.name = '小土豆';
obj.age = 18;

除此之外,在JavaScript中也可以通过构造函数自定义对象。

function Cat(){}
var catMimi = new Cat(); // 自定义对象

如果一个函数使用new关键字调用,那么这个函数就可以称为是构造函数,否则就是普通函数

什么是原型

一句话简单总结原型:原型是一个对象

在后面的总结中,原型可能会被描述为原型对象,其等价于原型

原型从哪里来?原型这个对象存在于哪里,需要通过代码去创建吗?

我们说对象是一系列属性的集合,那原型这个对象包含什么属性呢?

如何操作和使用原型?

接下来我们一个一个问题去探究。

原型从哪里来

JavaScript会为所有的函数创建一个原型

function Cat(){}

上面的代码中我们创建了一个Cat函数,那这个Cat函数就有一个原型,用代码表示就是:Cat.prototype

同样我们创建一个函数Fn1,函数Fn1就有一个原型,用代码表示就是Fn1.prototype

函数名称大写小写本质上没有任何区别

原型包含哪些属性

前面我们说过以下这两点:

  • 原型是一个对象
  • 对象是一系列属性的集合

原型都包含哪些属性呢?

前面我们已经知道原型用代码表示就是:functionName.prototype,那我们在代码中console.log一下。

function Cat(){}
console.log("Cat.prototype:");
console.log(Cat.prototype); function Dog(){}
console.log("Dog.prototype:");
console.log(Dog.prototype);

Firefox浏览器中的输出结果如下:

可以看到函数的原型默认有两个属性:constructor<prototype>

其中,函数原型的constructor属性指向函数本身。

函数原型的<propotype>属性称为隐式原型,后面我们会分出一节单独介绍隐式原型

如何操作和使用原型

正常我们操作一个普通对象的方式是下面这样的:

var obj = {};          // 创建对象
obj.name = '小土豆'; // 为对象添加属性
obj.age = 18; // 为对象添加属性
var name = obj.name; // 访问对象属性

原型既然也是一个对象,所以操作原型的方式和上述的方式相同。

function Cat(){}
Cat.prototype.type = 'cat';
Cat.prototype.color = 'White';
Cat.prototype.sayInfo = function(){
console.log(this.type + ' is ' + this.color);
}

此时再次打印Cat.prototype就能看到我们添加到原型上的属性:

访问原型对象上的方法和属性:

以上这些操作原型的方法,对于真正的项目开发并没有什么参考价值,不过不用着急,后面我们会详细讲解

隐式原型

前面我们在总结函数的原型对象时提到过隐式原型

那实际上,JavaScript会为所有的对象创建叫隐式原型的属性。我们一直说原型是一个对象,所以在上面的截图中,原型也有一个隐式原型属性。

隐式原型的代码表示

隐式原型是对象的私有属性,在代码中可以这样访问:obj.__proto__

obj.__proto__这种写法是非标准的,一些低版本的浏览器并不支持这样的写法

我们在浏览器的控制台中实际访问一下:

从打印的结果可以看到隐式原型也是一个对象,那隐式原型这个对象里面又包含什么属性呢?下面我们一起来看看。

隐式原型存在的意义

首先我们写一个简单的示例:

function Cat(){}
var catMimi = new Cat();
var catJuju = new Cat();

在上面这段代码中,我们创建了一个Cat函数,并且通过new关键字创建了以Cat构造函数的两个实例对象catMimicatJuju

接下来我们在浏览器的console工具中看看这两个实例对象的隐式原型都包含了那些属性。

可以看到,catMimi.__proto__catJuju._proto__的结果貌似是一样的,而且眼尖的同学应该也发现了这个打印结果似乎和前面一节【原型包含那些属性】中打印的Cat.prototype是一样的。

那话不多说,我们用==运算符判断一下即可:

可以看到所有的判断结果均为true

由于对象catMimicatJuJu都是由Cat函数创建出来的实例,所以总结出来结论就是:对象的隐式原型__proto__指向创建该对象的函数的原型对象

原型链:原型和隐式原型存在的意义

前面我们总结了原型隐式原型的概念以及如何使用代码操作原型隐式原型,总的看来原型隐式原型好像也没有特别厉害的地方,它们到底有什么用呢?

所有的实例对象共享原型上定义的属性和方法

我们来看下面这样一个示例:

function Cat(name, age){
this.type = 'RagdollCat'; //布偶猫
this.eyes = 2;
this.name = name;
this.age = age;
this.sayInfo = function(){
console.log(this.type + ' ' + this.name + ' is ' + this.age + ' years old');
}
}

在这个示例中,我们创建了一个Cat函数,同时Cat函数有五个属性:typeeyesnameagesayInfo,其中typeeyes属性已经有了初始值,而nameage通过参数传递并赋值;sayInfo对应是一个函数,打印出typenameage的值。

接着我们创建Cat的两个实例对象catMimicatJuju,并传入不同的nameage参数。

var catMimi = new Cat('Mimi', 1);
var catJuju = new Cat('Juju', 2);

控制台查看一下我们创建的对象:

可以看到这两个对象有着相同的属性,由于typeeyes是在Cat函数创建时已经有了固定的初始值,所以这两个属性值是相同的;sayInfo函数也都是相同的功能,打印出一些属性的信息;只有nameage是通过参数传递的,各自的值不相同。除此之外呢,catMimicatJuju是两个不同的对象,两者的属性值互相独立,修改其中任意一个的属性值并不会影响另外一个对象的属性值。

假如之后我们有更多这样的对象,JavaScript还是会为每一个对象创建相同的属性,而这些所有的对象都拥有着相同的typeeyes属性值和相同功能的sayInfo函数。这无疑造成了内存浪费,那这个时候我们就可以将这些属性定义到函数的原型对象上:

function Cat(name, age){
this.name = name;
this.age = age;
} Cat.prototype.type = 'RagdollCat'; //布偶猫
Cat.prototype.eyes = 2;
Cat.prototype.sayInfo = function(){
console.log(this.type + ' ' + this.name + ' is ' + this.age + ' years old');
}
var catMimi = new Cat('Mimi', 1);
var catJuju = new Cat('Juju', 2);

然后我们再来看看这两个对象:

可以看到这两个对象现在只包含了两个属性,就是Cat构造函数内容内部定义的两个属性:nameage

接着我们在去访问对象上的typeeyessayInfo

我们的实例对象还是可以正常访问到属性,方法也打印出来正确的信息。那到底是怎么访问到的呢?

原型链

在上一个示例代码中,我们将一些属性方法定义到函数的原型上,最后使用该函数创建出来的实例对象可以正常访问原型上定义的属性方法,这是怎么做到的呢?

前面我们说过:对象的隐式原型指向创建该对象的函数的原型对象,所以当实例对象中没有某个属性时,JavaScript就会沿着该实例对象隐式原型去查找,这便是我们所说的原型链

那既然是链,我们想到的应该是一个连着一个的东西,所以应该不仅仅是当前实例对象的隐式原型指向创建该对象的函数的原型对象,所以我们在对catMimi对象做点操作:

在上面的操作,我们调用了catMimihasOwnProperty方法,很明显我们并没有为这个对象定义该方法,那这个方法从哪里来呢?

答案依然是原型链

  • 调用catMimi.hasOwnProperty()方法
  • 在实例对象catMimi中查找属性,发现没有该属性
  • catMimi.__proto__中查找,因为catMimi.__proto__=Cat.prototype(实例对象的隐式原型指向创建该实例的函数的原型),也就是在Cat.prototype中查找hasOwnProperty属性,很明显Cat.prototype也没有该属性
  • 于是继续沿着Cat.prototype.__proto__查找,又因为Cat.prototype.__proto__ = Object.prototype(我们一直在强调原型是一个对象,既然是对象,就是由Object函数创建的,所以Cat.prototype隐式原型指向Object函数的原型)

我们打印一下Object.prototype的是否包含hasOwnProperty属性:

可以看到,Object.prototype中存在hasOwnProperty属性,所以catMimi.hasOwnPrototype实际上调用的是Object.prototype.hasOwnProperty

总结

本篇文章到此基本就基本结束了,相信大家应该对原型原型链有了一定的了解。最后呢,我们在对本篇文章做一个总结。

近期文章

详解Vue中的computed和watch

记一次真实的Webpack优化经历

JavaScript的执行上下文,真没你想的那么难

骨架屏(page-skeleton-webpack-plugin)初探

Vue结合Django-Rest-Frameword实现登录认证(二)

Vue结合Django-Rest-Frameword实现登录认证(一)

写在最后

如果这篇文章有帮助到你,️关注+点赞️鼓励一下作者

文章公众号首发,关注 不知名宝藏程序媛 第一时间获取最新的文章

笔芯️~

详解JavaScript中的原型的更多相关文章

  1. 详解JavaScript中的原型和继承-转自颜海镜大大

    本文将会介绍面向对象,继承,原型等相关知识,涉及的知识点如下: 面向对象与继承 CEOC OLOO 臃肿的对象 原型与原型链 修改原型的方式 面向对象与继承 最近学习了下python,还写了篇博文&l ...

  2. 详解Javascript中的原型与原型链

    目录 知识点 参考资料 结束语 知识点 面向对象编程 我们熟悉的Java和C#里,面向对象的两个基本概念是类class和实例instance,而ES6以前的Javascript并没有设计class. ...

  3. 详解javascript中的this对象

    详解javascript中的this对象 前言 Javascript是一门基于对象的动态语言,也就是说,所有东西都是对象,一个很典型的例子就是函数也被视为普通的对象.Javascript可以通过一定的 ...

  4. (转载)详解Javascript中prototype属性(推荐)

    在典型的面向对象的语言中,如java,都存在类(class)的概念,类就是对象的模板,对象就是类的实例.但是在Javascript语言体系中,是不存在类(Class)的概念的,javascript中不 ...

  5. 【转】详解JavaScript中的this

    ref:http://blog.jobbole.com/39305/ 来源:foocoder 详解JavaScript中的this JavaScript中的this总是让人迷惑,应该是js众所周知的坑 ...

  6. 详解 javascript中offsetleft属性的用法(转)

    详解 javascript中offsetleft属性的用法 转载  2015-11-11   投稿:mrr    我要评论 本章节通过代码实例介绍一下offsetleft属性的用法,需要的朋友可以做一 ...

  7. this详解:JAVASCRIPT中的this到底是谁?

    语法 this 全局对象 在全局执行上下文(函数之外),this引用的是全局对象. console.log(this.document === document); // true // In web ...

  8. 【转】详解JavaScript中的异常处理方法

    有三种类型的编程错误:(1)语法错误和(2)运行时错误(3)逻辑错误:语法错误: 语法错误,也被称为解析错误,在编译时进行传统的编程语言,并出现在JavaScript解释时. 例如,下面一行将导致一个 ...

  9. 详解JavaScript中的Event Loop(事件循环)机制

    前言 我们都知道,javascript从诞生之日起就是一门单线程的非阻塞的脚本语言.这是由其最初的用途来决定的:与浏览器交互. 单线程意味着,javascript代码在执行的任何时候,都只有一个主线程 ...

随机推荐

  1. Codeforces Global Round 8 D. AND, OR and square sum (贪心,位运算)

    题意:有\(n\)个数,选择某一对数使二者分别\(or\)和\(and\)得到两个新值,求操作后所有数平方和的最大值. 题解:不难发现每次操作后,两个数的二进制表示下的\(1\)的个数总是不变的,所以 ...

  2. Codeforces Round #527 (Div. 3) C. Prefixes and Suffixes (思维,字符串)

    题意:给你某个字符串的\(n-1\)个前缀和\(n-1\)个后缀,保证每个所给的前缀后缀长度从\([1,n-1]\)都有,问你所给的子串是前缀还是后缀. 题解:这题最关键的是那两个长度为\(n-1\) ...

  3. C# TCP应用编程二 同步TCP应用编程

    不论是多么复杂的TCP 应用程序,双方通信的最基本前提就是客户端要先和服务器端进行TCP 连接,然后才可以在此基础上相互收发数据.由于服务器需要对多个客户端同时服务,因此程序相对复杂一些.在服务器端, ...

  4. 7.PowerShell DSC之模式

    DSC两种模式 DSC有两种模式,Push模式和Pull模式 Push模式 基本流程 写配置--编译生成mof--推送到目标服务器,由目标服务器LCM执行mof并进行指定的配置 优点 架构简单.成本低 ...

  5. PHP的常用函数 持续更新

    PHP的常用函数 前言: 由于害怕遗忘,故在此记录下常用的php函数,以便复习 1 define函数 作用:定义常量 用法 <?php define('a',100); ?> 2 intv ...

  6. Dcoker命令使用详解

    Docker语法说明 docker [OPTIONS] COMMAND [arg...] OPTIONS --config=~/.docker :指定本地客户端配置文件. -D, --debug :开 ...

  7. Redis Cluster 分布式集群(下)

    Redis Cluster 搭建(工具) 环境准备 节点 IP 端口 节点① 172.16.1.121 6379,6380 节点② 172.16.1.122 6379,6380 节点③ 172.16. ...

  8. spring再学习之简单测试

    一.spring是怎么工作的那,通过一个类装载进容器进行描述: 首先创建一个类user: package cn.itcast.bean; public class User { public User ...

  9. 牛客网多校第9场 E Music Game 【思维+数学期望】

    题目:戳这里 题意:鼠标点击n下,第i次点击成功的概率为p[i],连续点击成功x次可以获得x^m分,求n次点击总分数的数学期望. 解题思路:数学期望的题很多都需要转化思维,求某一个单独状态对整体答案的 ...

  10. hdu1228双指针

    #include <iostream> #include <cstdio> #include <cstring> using namespace std; char ...