理解原型的几个关键点:

1、所有的引用类型(数组、函数、对象)可以自由扩展属性(除null以外);

2、所有的引用类型(对象)都有一个’_ _ proto_ _'属性(也叫隐式原型,它是一个普通的对象),指向原型对象;

3、所有的函数都有一个’prototype’属性(这也叫显式原型,它也是一个普通的对象,该对象就是函数的原型对象,对象中包含所有实例对象可以共享的属性和方法)。’prototype’属性是函数独有的,任何函数在创建的时候,其实会默认同时创建该函数的prototype对象;

4、所有引用类型,它的’_ _ proto_ _'属性指向它的构造函数的’prototype’属性(所以  函数._ _ proto_ _ ===function.prototype);

5、当试图得到一个对象的属性时,如果这个对象本身不存在这个属性,那么就会去它的’_ _ proto_ _'属性(也就是它的构造函数的’prototype’属性)中去寻找。

6、constructor属性也是对象才拥有的,指向该对象的构造函数。函数创建的对象.__proto__ === 该函数.prototype,该函数.prototype.constructor===该函数本身,故通过函数创建的对象即使自己没有constructor属性,它也能通过__proto__找到对应的constructor,所以任何对象最终都可以找到其构造函数(null如果当成对象的话,将null除外)

原型:

先来看一个原型的例子。

     //这是一个构造函数function Foo(name,age){this.name=name;this.age=age;}/*根据要点3,所有的函数都有一个prototype属性,这个属性是一个对象再根据要点1,所有的对象可以自由扩展属性于是就有了以下写法*/Foo.prototype={// prototype对象里面又有其他的属性showName:function(){console.log("I'm "+this.name);//this是什么要看执行的时候谁调用了这个函数},showAge:function(){console.log("And I'm "+this.age);//this是什么要看执行的时候谁调用了这个函数}}var fn=new Foo('小明',19)/*当试图得到一个对象的属性时,如果这个对象本身不存在这个属性,那么就会去它构造函数的'prototype'属性中去找*/fn.showName(); //I'm 小明fn.showAge(); //And I'm 19

这就是原型,很好理解。那为什么要使用原型呢?

试想如果我们要通过Foo()来创建很多很多个对象,如果我们是这样子写的话:

 function Foo(name,age){this.name=name;this.age=age;this.showName=function(){console.log("I'm "+this.name);}this.showAge=function(){console.log("And I'm "+this.age);}}

那么我们创建出来的每一个对象,里面都有showName和showAge方法,这样就会占用很多的资源。
而通过原型来实现的话,只需要在构造函数里面给属性赋值,而把方法写在Foo.prototype属性(这个属性是唯一的)里面。这样每个对象都可以使用prototype属性里面的showName、showAge方法,并且节省了不少的资源。


原型链

理解了原型,那么原型链就更好理解了。

#####下面这段话可以帮助理解原型链
根据要点5,当试图得到一个对象的属性时,如果这个对象本身不存在这个属性,那么就会去它构造函数的’prototype’属性中去寻找。那又因为’prototype’属性是一个对象,所以它也有一个’_ _ proto_ _'属性。

     // 构造函数function Foo(name,age){this.name=name;this.age=age;}Object.prototype.toString=function(){//this是什么要看执行的时候谁调用了这个函数。console.log("I'm "+this.name+" And I'm "+this.age);}var fn=new Foo('小明',19);fn.toString(); //I'm 小明 And I'm 19console.log(fn.toString===Foo.prototype.__proto__.toString); //trueconsole.log(fn.__proto__ ===Foo.prototype)//trueconsole.log(Foo.prototype.__proto__===Object.prototype)//trueconsole.log(Object.prototype.__proto__===null)//true

是不是觉得有点奇怪?我们来分析一下。

首先,fn的构造函数是Foo()。所以:
fn._ _ proto _ _=== Foo.prototype
又因为Foo.prototype是一个普通的对象,它的构造函数是Object,所以:
Foo.prototype._ _ proto _ _=== Object.prototype
通过上面的代码,我们知道这个toString()方法是在Object.prototype里面的,当调用这个对象的本身并不存在的方法时,它会一层一层地往上去找,一直到null为止。

所以当fn调用toString()时,JS发现fn中没有这个方法,于是它就去Foo.prototype中去找,发现还是没有这个方法,然后就去Object.prototype中去找,找到了,就调用Object.prototype中的toString()方法。

__proto__属性的作用就是当访问一个对象的属性时,如果该对象内部不存在这个属性,那么就会去它的__proto__属性所指向的那个对象(父对象)里找,一直找,直到__proto__属性的终点null,再往上找就相当于在null上取值,会报错。通过__proto__属性将对象连接起来的这条链路即我们所谓的原型链。

另外,在使用原型的时候,一般推荐将需要扩展的方法写在构造函数的prototype属性中,避免写在_ _ proto _ _属性里面。

总结:

  • 当所有的实例对象都需要共享属性和方法时,通过原型来实现就是将属性方法放在实例对象的构造函数的prototype属性中(该属性值就是原型对象,包含共享属性和方法);
  • 访问一个对象的属性时,先在基本属性中查找,如果没有,再沿着__proto__这条链向上找,这就是原型链。
  • 根据原型链可以确定继承关系。由于所有的对象的原型链都会找到Object.prototype,因此所有的对象都会有Object.prototype的方法。这就是所谓的“继承”。
  • 对象引用类型通过instanceof来判断。

版权声明:文章内容主要综合来自链接处,

https://blog.csdn.net/qq_36996271/article/details/82527256

https://blog.csdn.net/cc18868876837/article/details/81211729

JS中的prototype、__proto__与constructor,原型和原型链相关推荐

  1. JS中的prototype、__proto__与constructor

    作为一名前端工程师,必须搞懂JS中的prototype.__proto__与constructor属性,相信很多初学者对这些属性存在许多困惑,容易把它们混淆,本文旨在帮助大家理清它们之间的关系并彻底搞 ...

  2. JS中的prototype、__proto__与constructor(图解)

    作为一名前端工程师,必须搞懂JS中的prototype.__proto__与constructor属性,相信很多初学者对这些属性存在许多困惑,容易把它们混淆,本文旨在帮助大家理清它们之间的关系并彻底搞 ...

  3. 帮你彻底搞懂JS中的prototype、__proto__与constructor(图解)

    帮你彻底搞懂JS中的prototype.__proto__与constructor(图解) 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明. 本文 ...

  4. 这一篇彻底搞懂JS中的prototype、__proto__与constructor真的很好

    文章目录 1. 前言 2. _ _ proto _ _ 属性 3. prototype属性 4. constructor属性 5. 总结 提示:不要排斥,静下心来,认真读完,你就搞懂了!(可以先看一下 ...

  5. (转)帮你彻底搞懂JS中的prototype、__proto__与constructor(图解)

    文章目录 1. 前言 2. _ _ proto _ _ 属性 3. prototype属性 4. constructor属性 5. 总结 提示:不要排斥,静下心来,认真读完,你就搞懂了!(可以先看一下 ...

  6. JS中的prototype

    JS中的prototype 2011-06-03 14:40 by 轩脉刃, 17040 阅读, 16 评论, 收藏, 编辑 JS中的phototype是JS中比较难理解的一个部分 本文基于下面几个知 ...

  7. js中的prototype的理解

    2019独角兽企业重金招聘Python工程师标准>>> JS中的prototype是JS中比较难理解的一个部分 1 原型法设计模式 在.Net中可以使用clone()来实现原型法 原 ...

  8. JS中对于prototype的理解

    JS中的prototype是JS中比较难理解的一个部分 本文基于下面几个知识点: 1 原型法设计模式 在.Net中可以使用clone()来实现原型法 原型法的主要思想是,现在有1个类A,我想要创建一个 ...

  9. JS中ptototype和__proto__的关系

    学到原型的时候感觉头都大了/(ㄒoㄒ)/~~ 尤其是ptototype和__proto__ 傻傻分不清  通过多番查找资料,根据自己的理解,总结如下: 一.构造函数: 构造函数:通过new关键字可以用 ...

最新文章

  1. 新疆弃光量下降14% 弃光问题仍然难解
  2. JVM锁和分布式锁是什么关系
  3. php解决链接 amp,php处理替换链接参数
  4. 埃斯顿驱动器参数设置_驱动器参数设置讲解(伺服步进)
  5. CPU-内存-IO-网络调优
  6. thinkphp __PUBLIC__的定义 __ROOT__等常量的定义
  7. 解决ImageLoader加载HTTPS图片证书校验异常问题
  8. Git提交空文件夹的技巧
  9. 01_决策树案例一:鸢尾花数据分类
  10. Redis学习笔记之入门基础知识——其他特性
  11. Data crossstore between Mongo and JPA
  12. Linux部分命令使用说明
  13. 【SketchUp插件】10款吊炸天的SU插件,拿走不谢!
  14. 数据仓库是如何分层的?
  15. 成都市计算机会考,四川省高中信息技术会考资料及试题
  16. 使用Charles进行HTTPS抓包
  17. 王天官系古盐山县(今孟村县王帽圈)人
  18. 外汇投资风险在哪里.
  19. esxi 虚拟机的控制台上键盘无法输入
  20. RGB/YUV/YIQ 颜色空间

热门文章

  1. mysql 聚合函数 怎么用在条件里_MySql 中聚合函数增加条件表达式的方法
  2. [转载] 微服务安全和治理
  3. [转载] python循环中break、continue 、exit() 、pass的区别
  4. lcase和ucase_在SQL中使用UCASE(),LCASE()和MID()函数
  5. 2020知道python语言应用答案_2020知到Python语言应用答案章节期末答案
  6. nextboolean()_Java Random nextBoolean()方法与示例
  7. Java ByteArrayInputStream skip()方法与示例
  8. java 方法 示例_Java集合的lastlastIndexOfSubList()方法和示例
  9. 单线程的Redis为什么却能支撑高并发?
  10. int?id与id??1 的意思