问题

初学js的同学,总是搞不清楚js中的原型是什么东西,看着控制台打印出来的一串串__proto__,迷惑不已。

例如我定义一个Person,创建一个实例p,并打印实例。

function Person(){}
var p = new Person();
console.log(p)



图中,打印出来一个Person的实例对象p,

这个对象有个__proto__ 属性,这个是什么东西?

__proto__属性下又有constructor属性和__proto__属性。

constructor是什么,为什么打印结果是 Person()?

另外一个__proto__是什么?

为了解答上边的问题,我们需要了解一些相关概念。

一、原型

prototype

首先,说明一下,JS中,万物皆对象。

每个函数对象都有一个属性prototype(函数对象特有属性),这个属性是一个引用,指向一个对象,这个对象的作用就是包含所有实例共享的属性和方法。我们把这个对象就叫做原型对象,也叫显式原型

__proto__

每个对象都有一个属性__proto__,也可称为隐式原型,对象的隐式原型指向创建该对象的构造函数的原型(prototype)。

constructor

函数的原型对象有一个constructor属性,这个属性是一个引用,用于指向原构造函数。

关系

我们怎么理解上边的概念,而它们之间又有什么联系呢?

我们不妨打印出来p.__proto__属性:



可以看到,打印出来是一个对象,对象里边有个属性constructor。

constructor是什么呢,打印如下:



可以看到,打印出来是函数,代表的就是Person构造函数本身.

再依次打印出来Person构造函数的原型Person.prototype,

和其原型的属性constructor:



从上图我们就可以得出:

1.实例对象p有属性__proto__ 指向的就是创建它的构造函数的原型对象Person.prototype。

2.构造函数的原型对象Person.prototype的属性constructor指向Person构造函数本身。

我们可以验证一下:

但是,可能又有些同学有疑惑了,那既然所有对象都有__proto__属性,

那构造函数Person()的属性__proto__指向谁呢?

当然是指向它的构造函数的原型对象了。

函数的构造函数就是Function(),因此这里的__proto__指向Function.prototype。

那原型对象也是对象,它的__proto__属性指向谁呢?

同理,指向它的构造函数的原型对象,即Object.prototype。

这里,我们就不得不提一下,原型链的概念了。

原型链

原型链是一种机制,指的是js中,每个对象都有一个属性__proto__,指向它的构造函数的原型对象。原型对象也是一个对象,因此也有__proto__属性指向原型对象的原型对象,这样一层层向上直到对象的原型对象为空(Object的原型对象Object.prototpye的属性__proto__为null)。

因此,例子中原型链的关系如下:

p.__proto__ 指向 Person.prototype,

Person.prototype.__proto__指向的就是Object.prototype,

Object.prototpye.__proto__ 指向null

为了理解上边例子的原型链关系,我们画一张图来增加理解:



至此,上边的问题是不是心里已经有答案了呢。

总结

1.每个对象都有一个__proto__属性,指向创建它的构造函数的原型对象。

作用:构成原型链,用于实现基于原型的继承。

2.函数除了有__proto__属性,还有一个prototype属性,用来指向函数的原型对象。

作用:用于实现基于原型的继承和属性共享。

因此,你会在很多地方见到有类似这样的写法:

function Person(){}
Person.prototype.hello = function(){
console.log("hello")
}

上边代码表示,所有Person创建的实例对象都可以共享hello方法。

3.构造函数的原型对象有属性constructor,指向构造函数本身。

PS:欢迎关注公众号:「如若清风」,一起交流学习。

JavaScript中的原型prototype和__proto__的区别及原型链概念的更多相关文章

  1. javascript的构造函数和实例对象、prototype和__proto__的区别,原型对象及构造器的理解

    一.前言 我们先通过代码来分别打印出实例对象.构造函数,以及修改了原型对象的构造函数,通过对比内部结构来看看他们之间的区别. //定义构造函数 function Person(name, age){ ...

  2. JS 中的原型 -- prototype、__proto__ 以及原型链

    原文: 1.深入理解javascript原型和闭包——prototype原型 2.三张图搞懂JavaScript的原型对象与原型链 打开浏览器控制台,任意定义一个对象,打印出来后,会发现有最后一定有一 ...

  3. 前端开发:面向对象与javascript中的面向对象实现(二)构造函数与原型

    前端开发:面向对象与javascript中的面向对象实现(二)构造函数与原型 前言(题外话): 有人说拖延症是一个绝症,哎呀治不好了.先不说这是一个每个人都多多少少会有的,也不管它究竟对生活有多么大的 ...

  4. JavaScript中的Array.prototype.slice.call()方法学习

    JavaScript中的Array.prototype.slice.call(arguments)能将有length属性的对象转换为数组(特别注意: 这个对象一定要有length属性). 但有一个例外 ...

  5. javascript中apply、call和bind的区别,容量理解,值得转!

    a)  javascript中apply.call和bind的区别:http://www.cnblogs.com/cosiray/p/4512969.html b)  深入浅出 妙用Javascrip ...

  6. javascript中三目运算符和if else有什么区别

    javascript中三目运算符和if else有什么区别今天写了一个图片轮播的小demo,用到了判断先试了一下if else,代码如下:if(n >= count-1){n =0;}else{ ...

  7. JavaScript中var和this定义变量的区别

    JavaScript中var和this定义变量的区别 在js中声明变量时可以使用var和this,但使用this的有很大一部分参考书是没有的,经过查阅相关资料总结如下: 用var和this声明变量,存 ...

  8. javascript中back(-1)和go(-1)的区别

    javascript中back(-1)和go(-1)的区别 一.总结 一句话总结: 数据 history.back(-1):直接返回当前页的上一页,数据全部消息,是个新页面 history.go(-1 ...

  9. JavaScript中基本数据类型和引用数据类型的区别(栈——堆)

    JavaScript中基本数据类型和引用数据类型的区别 1.基本数据类型和引用数据类型 ECMAScript包括两个不同类型的值:基本数据类型和引用数据类型. 基本数据类型指的是简单的数据段,引用数据 ...

随机推荐

  1. 计蒜客 39280.Travel-二分+最短路dijkstra-二分过程中保存结果,因为二分完最后的不一定是结果 (The 2019 ACM-ICPC China Shannxi Provincial Programming Contest M.) 2019ICPC西安邀请赛现场赛重现赛

    Travel There are nn planets in the MOT galaxy, and each planet has a unique number from 1 \sim n1∼n. ...

  2. 原创:Spark中GraphX图运算pregel详解

    由于本人文字表达能力不足,还是多多以代码形式表述,首先展示测试代码,然后解释: package com.txq.spark.test import org.apache.spark.graphx.ut ...

  3. vue+element 表格导出Excel文件

    https://www.cnblogs.com/bobodeboke/p/8867481.html  非常感谢 这个大佬 才让我搞到了Blob.js 和 Export2Excel.js 如果最后运行时 ...

  4. vue data不可以使用箭头函数的问题解析

    这篇文章主要介绍了vue data不可以使用箭头函数问题,本文通过源码解析给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下 首先需要明确,a() {}和 b: () => {} ...

  5. 微信小程序之页面导航栏

    效果图: 页面有点丑,作为初次学习,页面可以要求不那么美观,先学会再说.毕竟后面可以优化的很漂亮. 代码实例如下: <view class="section btn-area" ...

  6. uniapp - 阿里图库字体图标使用

    [iconfont下载] https://www.iconfont.cn/search/index?searchType=icon&q=%E4%B8%8A%E4%BC%A0 可能报错,找不到线 ...

  7. Real-time Multiple People Tracking with Deeply Learned Candidate Selection and Person Re-identification

    Real-time Multiple People Tracking with Deeply Learned Candidate Selection and Person Re-identificat ...

  8. python pycharm错误集锦

    url:http://www.cnblogs.com/hinimix/p/8016859.html 1, this list creation could be rewritten as a list ...

  9. jmeter BeanShell断言(四)

    Bean Shell常用内置变量 JMeter在它的BeanShell中内置了变量,用户可以通过这些变量与JMeter进行交互,其中主要的变量及其使用方法如下: log:写入信息到jmeber.log ...

  10. C#发起Http请求,调用接口

    //方法1. Post 异步请求,普通的异步请求,传输普通的字符串等,对于有html代码的字段值的传输支持不好,如果需要传输html,二进制等数据的传输,请使用下面第二个方法,即使用UploadDat ...