深入__proto__和prototype的区别和联系
前话
有一个一个装逼的同事,写了一段代码
function a(){}
a.__proto__.__proto__.__proto__
然后问我,下面这个玩意a.__proto__.__proto__.__proto__是啥,然后我一脸懵逼,prototype还知道一点,这个__proto__,还来三个,是个什么鬼。于是我一直不能放下这个问题,虽然我很懒,很不喜欢费脑子,但是这个坎还是过不去,最近两天研究了大半天,就有了这篇文章。
我先说出答案, 上面的值为 null。我还很负责的告诉你,下面的_a.__proto__.__proto__.__proto__也是null
function a(){}
var _a = new a();
_a.__proto__.__proto__.__proto__
先来一张非常经典的图,真的是非常经典,你看懂他,你就懂了整个世界,然后整个世界就等着你去拯救整个世界。
)
正文之前,__proto__和prototype
都谁有的问题
typeof === object的有__proto__ , null和undefined都没有
typeof === function的有__proto__和prototype
__proto__ 是什么
__proto__ 一般情况指向的是该对象的构造函数的prototype,一般情况,因为还有很二般的情况。
先来看个简单的例子, 下面的输出是true
function a(){}
var _a = new a()
console.log(_a.__proto__ === a.prototype)
那我问_a.__proto__.__proto__为什么呢,你会这么推导么,
依据上面_a.__proto__ === a.prototype,那么_a.__proto__.__proto__就等同a.prototype.__proto__ , 那么我们就再推到等于 a.prototype.constructor.prototype,然后你去一比,结果是false。
_a.__proto__.__proto__ === a.prototype.constructor.prototype
// false
几条规则
这个先不纠结, 我们先看看上图,我们先得知道或者记住这几个规则
Object.prototype.__proto__ === null
不要纠结,铁律Object.__proto__ === Function.prototype
Object,Number, Error等等这些函数都是Function创建的,下面就说明
这些的constructor就是Function,这里比较有意思的就是 Function.constructor也是Function。
那就有Object.__proto__ === Function.prototype === Function.__proto__
Object.constructor.prototype === Function.prototype
// true
Function.constructor === Function
// true
Function.prototype.__proto__ === Object.prototype
这个就是这样的设计,
Function.prototype.constructor === Object
// false
- 自定义函数,没有修改默认原型的情况下,比如
function Person(){},
Person.prototype.__proto__ === Object.prototype
Person.constructor === Function
对比3一看,Function和构造函数是Function的Person,他们prototype.__proto__的指向是一样的。
进入正题
有这几个基本东西,我们就可以来推导了。
先看下面的代码,
js 我们来推到 aaa.__proto__.__proto__.__proto__
function aaa(){}
var _aaa = new aaa()
aaa.__proto__
aaa构造函数是Function
aaa.constructor === Function
aaa.__proto__ === Function.prototypeaaa.__proto__.__proto__
aaa.__proto__.__proto__ === Function.prototype.__proto__
依据Function.prototype.__proto__ === Object.prototype
aaa.__proto__.__proto__ === Function.prototype.__proto__ === Object.prototypeaaa.__proto__.__proto__.__proto__
aaa.__proto__.__proto__.__proto__ === Object.prototype.__proto__
依据Object.prototype.__proto__ === null
aaa.__proto__.__proto__.__proto__ === null
还是上面代码,我们接着推导_aaa.__proto__.__proto__.__proto__
_aaa.__proto__
_aaa的构造函数是 aaa
_aaa.constructor === aaa
_aaa.__proto__ === _aaa.constructor.prototype
_aaa.__proto__ === aaa.prototype_aaa.__proto__.__proto__
_aaa.__proto__.__proto__ === aaa.prototype.__proto__
参考图,Foo.prototype.__proto__ === Object.prototype
_aaa.__proto__.__proto__ === aaa.prototype.__proto__ === Object.protype_aaa.__proto__.__proto__.__proto__
_aaa.__proto__.__proto__.__proto__ === Object.protype.__proto__
依据Object.prototype.__proto__ === null
_aaa.__proto__.__proto__ === null
正文延伸, 加上继承关系
我们再来看看,带继承关系的
function aaa(){}
function bbb(){}
bbb.prototype = new aaa()
var _bbb = new bbb();
bbb.__proto__.__proto__.__proto__ === null
这个没啥好说,
关键来看看 bbb.prototype.__proto__.__proto__.__proto__
bbb.prototype.__proto__
bbb.prototype.__proto__ === bbb.prototype.constructor.prototype
bbb.prototype的原型是 aaa的实例, bbb原型的构造函数就是aaa,所以
bbb.prototype.__proto__ === aaa.prototypebbb.prototype.__proto__.__proto__
bbb.prototype.__proto__.__proto__ === aaa.prototype.__proto__
参考图,Foo.prototype.__proto__ === Object.prototype
bbb.prototype.__proto__.__proto__ === Object.prototypebbb.prototype.__proto__.__proto__
bbb.prototype.__proto__.__proto__ .__proto__=== Object.prototype.__proto__ === null
再来看看_bbb.__proto__.__proto__.__proto__ .__proto__
_bbb.__proto__
_bbb.__proto__ === bbb.prototype- _bbb.proto.proto
_bbb.__proto__.__proto__ === bbb.prototype._proto__ === bbb.prototype.constructor.prototype === aaa.prototype - _bbb.proto.proto.proto
_bbb.__proto__.__proto__.__proto__ === aaa.prototype.__proto__
参考图Foo.prototype.__proto__ === Object.prototype
_bbb.__proto__.__proto__.__proto__ === aaa.prototype.__proto__ === Object.prototype _bbb.__proto__.__proto__.__proto__.__proto__
_bbb.__proto__.__proto__.__proto__.__proto__ === Object.prototype.__proto__ === null
正文 再加量
看看如下代码
function aaa(){}
var _aaa = new aaa()
function bbb(){}
bbb.prototype = new aaa();
var _bbb = new bbb();
function ccc(){}
ccc.prototype = new bbb()
var _ccc = new ccc()
我们再来分析_ccc的prototype和__proto__,你们会说,你有完没完
,那我就不分析了,我来推断:
推断:
任何自定义的function本身,三次
__proto__必然是null,也就是往上找三代
包括Function,Object, Error等等
Fucntion.proto 看图,依据
Object.__proto__ === Function.prototype === Function.__proto__
我们来推导Function.__proto__.__proto__ .__proto__
第一步:Function.__proto__ === Function.prototype
第二步:Function.__proto__.__proto__ === Function.prototype.__proto__ === Object.protetype
第三步:Function.__proto__.__proto__ .__proto__ === Object.protetype.__proto__ === null
都是Function构造出来的
我们来测试一下cccccc.__proto__.__proto__.__proto__ === null // true
继承关系的function fn,假设继承次数为n,
_fn = new fn();
那么_fn.__protot__[3 + n] === null
_ccc应该是3+2就是5次_ccc.__proto__.__proto__.__proto__.__proto__.__proto__ === null // true
继承关系的function fn,假设继承次数为n
推到fn.prototype.__proto__[3+n-1]
ccc应该是 4次__proto__ccc.prototype.__proto__.__proto__.__proto__.__proto__ === null // true
当然上面关联的关系,就自己慢慢看吧
正文之外, class
下面的代码也是遵守规则,至于为什么,问自己喽。
class aaa {}
class bbb extends aaa{}
class ccc extends bbb{};
var _ccc = new ccc()
关于Number,Boolen, String,Function, Date, Array, RegExp等的__proto__和prototype.proto`
__proto__
因为这些都是Function创建出来的函数,__proto__在函数上时就是表示构造函数的prototype,所以
.__proto__ === .constrcutor.prototype === Function.prototype.prototype.__proto__这些老骨头不遵循__proto__为构造函数的prototype
在上面提到过了,Function.prototype.__proto__ === Object.prototype,
类推,这些内置的老骨头的.prototype.__proto__ === Object.prototype
总结
总结, 特别需要记忆的:
Object.prototype.__proto__ === nullFunction.prototype.__proto__ === Object.prototype
内置Number,Boolen, String,Function, Date, Array, RegExp等一样Object.__proto__ === Function.prototype === Function.__proto__
联系2,这些东西都是Function创建出来的Math, JSON的__ptoto__是 Object.prototype
typeof 可以看出来这两个是object,而不是Function- function a(){} 这样创建出来,没有继承关系的函数
a.prototype.__proto__ === Object.prototype - 有继承关系的function看上面的推断
- 对象字面量和new Object() 比如, var a ={}, b = new Object(), c = [];
a.__proto__ === a.constructor.prototype === Object.protype
a.__proto__.__proto__ === Object.protype.__proto__ === null - 基本数据类型string,number,boolean,比如 var a = '', b=10, c= false,
b.__proto__ === b.constructor.prototype === Number.prototype
b.__proto__.__proto__ === Number.prototype.__proto__ === Object.prototype
b.__proto__.__proto__.__proto__ === Object.prototype.__proto__ === null - null和undefined没有
__proto__
最终
- 看图
- 浏览器输入
xx.__proto__或者xx.prototype自己看去
深入__proto__和prototype的区别和联系的更多相关文章
- 谈谈__proto__和prototype的区别
我想javascript中的原型链一直想很多初学javascript的同学感到非常的困惑,今天看了一些文章,结合自己的理解,把原型链这个东西从新来整理一下,如有不对之处,望大家帮忙指出. 首先,我们应 ...
- js中__proto__和prototype的区别和关系? 这样好理解多了
原型的概念 真正理解什么是原型是学习原型理论的关键.很多人在此产生了混淆,没有真正理解,自然后续疑惑更多. 首先,我们明确原型是一个对象,其次,最重要的是, Every function has a ...
- js中__proto__和prototype的区别和关系?
_proto__(隐式原型)与prototype(显式原型)1.是什么 显式原型 explicit prototype property: 每一个函数在创建之后都会拥有一个名为prototype的属性 ...
- Javascript深入__proto__和prototype的区别和联系
有一个一个装逼的同事,写了一段代码 function a(){} a.__proto__.__proto__.__proto__ 然后问我,下面这个玩意a.__proto__.__proto__.__ ...
- 理解js中__proto__和prototype的区别和关系
首先,要明确几个点:1.在JS里,万物皆对象.方法(Function)是对象,方法的原型(Function.prototype)是对象.因此,它们都会具有对象共有的特点.即:对象具有属性__proto ...
- [转载]js中__proto__和prototype的区别和关系
首先,要明确几个点:1.在JS里,万物皆对象.方法(Function)是对象,方法的原型(Function.prototype)是对象.因此,它们都会具有对象共有的特点.即:对象具有属性_ ...
- 【JavaScript】__proto__和prototype的区别和联系【整理】
var person={name:'ninja'}; person.prototype.sayName=function(){ return this.name; } Chrome运行结果: 提示找不 ...
- js中__proto__和prototype的区别和关系
首先,要明确几个点:1.在JS里,万物皆对象.方法(Function)是对象,方法的原型(Function.prototype)是对象.因此,它们都会具有对象共有的特点.即:对象具有属性_ ...
- (转)js中__proto__和prototype的区别和关系
作者:doris链接:https://www.zhihu.com/question/34183746/answer/58155878来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请 ...
随机推荐
- php之isset empty is_null的区别
isset:当前变量没有设置(即不存在),或者变量设置为null的时候,返回true,设置为“”或者0 都是返回的true empty:变量不存在,设置值为null,设置为“”,设置为0 都返回tru ...
- LCS 最长公共子子串
与 最长公共子序列类似 只是最长公共子串必须连续 所以只能走斜线!!!! ''' LCS 最长公共子序列 ''' def LCS_len(x, y): m = len(x) n = len(y) dp ...
- uva672
Gangsters N gangsters are going to a restaurant. The i-th gangster comes at the time Ti and has t ...
- 【4】Python对象
本章主题 Python对象 内建类型 标准类型操作符 值的比较 对象身份比较 布尔类型 标准类型内建函数 标准类型总览 各种类型 不支持的类型 Python对象 Python使用 ...
- jquery 添加列
{field:'action',title:'操作',width:70,align:'center', formatter:function(value,row,index){ if (row.e ...
- linux 异常
1. NoRouteToHostException异常问题的原因及解决 (转自:http://performtest163.blog.163.com/blog/static/1400769642011 ...
- Django使用本地css/js文件
Django使用本地css/js文件 在manager.py同层级下创建static文件夹, 里面放上css , js, images等文件或者文件夹 我的文件夹层级 然后只需在settings.py ...
- kylin构建cube优化
前言 下面通过对kylin构建cube流程的分析来介绍cube优化思路. 创建hive中间表 kylin会在cube构建的第一步先构建一张hive的中间表,该表关联了所有的事实表和维度表,也就是一张宽 ...
- 广播地址设为自己的ip地址,子网掩码设为255.255.255.255 是什么意思
最近看<构建高性能web站点>(老版)第12章讲LVS-DR的配置,有一段针对Real Server的配置不明所以: ifconfig lo:0 125.12.12.77 broadcas ...
- install ros-indigo-tf2
sudo apt-get install ros-indigo-tf2