彻底搞懂js原型与原型链
原生的原型链
function fn(){},fn 为 Function的一个实例,原型链为 null -> Object -> Function -> fn:
fn.__proto__ === Function.prototype
Function.prototype.__proto === Object.prototype
Object.prototype.__proto__ === null
原型链通过原型(prototype)链接,prototype 就是一个 只有(构造)函数才有的特殊的实例,包含了constructor指针(和__proto__,所有实例都有)
之后new出来的普通实例(非prototype)都包含这个prototype(引用名为__proto__):console.log(sonA) // {/* 其他属性 */ __proto__: Son.prototype}
举例来说,对于函数实例,所有函数实例.__proto__ === Function.prototype (通过这个能拿到Function上的通用属性,如bind函数)
甚至Function.__proto__ === Function.prototype, Object.__proto__ === Function.prototype
Function instanceof Object // true Function.__proto__.__proto__ === Object.prototype (优先是更具体的函数,然后才作为更抽象的对象,对象永远在原型链较上层)
Object instanceof Function // true Object.__proto__ = Function.prototype
原型链继承:
未继承时
console.log(Son.prototype) // { constructor: Son, __proto__: Object.prototype }
原型链继承了Father时,
console.log(Son.prototype) // { __proto__: Father.prototype }父类的一个实例, 通过new Father 或者 Object.create(Father.prototype)生成
需要手动赋予constructor:
Object.defineProperty(Son.prototype, "constructor", {
value: Son,
writable: false
});
console.log(Object.prototype)
{ __proto__: null }
原型链继承,本质就是修改函数的prototype,使其从指向Function变为指向父类;父类的对象刚好符合这个特性
巩固一下上面的知识点,请判断以下比较是否为真:
Object.prototype.__proto__ === null
Object.__proto__ === Function.prototype
Function.__proto__ === Function.prototype
fn.__proto___ === Function.prototype
Object.prototype === Function.prototype.__proto__
Object.prototype === fn.prototype.__proto__
Object.prototype.isPrototypeOf(fn)
Object.prototype.isPrototypeOf(fn.prototype)
Function.prototype === fn.__proto__;
Function.prototype.isPrototypeOf(fn)
Function.prototype.isPrototypeOf(fn.prototype) // false
答案:以上判断若未特殊说明,结果都为真
谈一下new
自定义原型链:基于原型链的继承
Father() {} -> Son() {]
继承的本质是,如何使Son的实例能访问到Father定义的属性
通过原型链,我们将Son的原型指向Father
Son原型链现在是指向的Function,Son.__proto__ === Function.prototype
我们希望Son.__proto__ === Father.prototype
按es5之前的规范,我们不能直接访问__proto__,那么只能修改prototype (ES5中用Object.getPrototypeOf函数获得一个对象的[[prototype]]。ES6中,使用Object.setPrototypeOf可以直接修改一个对象的[[prototype]])
我们知道 Father的实例的__proto__指向Father.prototype,那么Son.prototype等于Father的实例就可以
Son.prototype = new Father() 或者
Son.prototype = Object.create(Father.prototype)
这里有一个问题,prototype的constructor属性应该指向构造函数,而这种写法的constructor通过原型链会找到Father,所以需要定义一下constructor
Object.defineProperty(Son.prototype, "constructor", {
value: Son,
writable: false
});
顺便了解一下原型链查找顺序
优先查找对象的,对象上没有则通过__proto__找prototype,还没找到,就找prototype.__proto__, 也就是父类的prototype
比如
对象objA查找hasOwnProperty
objA -> Son.prototype -> Object.prototype -> Object.prototype.hasOwnProperty
函数funcA查找bind
funcA -> Function.prototype -> Function.prototype.bind
PS:对于 const fn = () => {} 这样的箭头函数,不属于原型链的范畴 可以参考 http://www.cnblogs.com/mengff/p/9656486.html
彻底搞懂js原型与原型链的更多相关文章
- 帮你彻底搞懂JS中的prototype、__proto__与constructor(图解)
作为一名前端工程师,必须搞懂JS中的prototype.__proto__与constructor属性,相信很多初学者对这些属性存在许多困惑,容易把它们混淆,本文旨在帮助大家理清它们之间的关系并彻底搞 ...
- 彻底搞懂 JS 中 this 机制
彻底搞懂 JS 中 this 机制 摘要:本文属于原创,欢迎转载,转载请保留出处:https://github.com/jasonGeng88/blog 目录 this 是什么 this 的四种绑定规 ...
- 让你彻底搞懂JS中复杂运算符==
让你彻底搞懂JS中复杂运算符== 大家知道,==是JavaScript中比较复杂的一个运算符.它的运算规则奇怪,容易让人犯错,从而成为JavaScript中“最糟糕的特性”之一. 在仔细阅读了ECMA ...
- 一文搞懂 js 中的各种 for 循环的不同之处
一文搞懂 js 中的各种 for 循环的不同之处 See the Pen for...in vs for...of by xgqfrms (@xgqfrms) on CodePen. for &quo ...
- 一文读懂JS中的原型和原型链(图解)
讲原型的时候,我们应该先要记住以下几个要点,这几个要点是理解原型的关键: 1.所有的引用类型(数组.函数.对象)可以自由扩展属性(除null以外). 2.所有的引用类型都有一个’_ _ proto_ ...
- 彻底搞懂js __proto__ prototype constructor
在开始之前,必须要知道的是:对象具有__proto__.constructor(函数也是对象固也具有以上)属性,而函数独有prototype 在博客园看到一张图分析到位很彻底,这里共享: 刚开始看这图 ...
- 一张图搞懂 Javascript 中的原型链、prototype、__proto__的关系 转载加自己的总结
1. JavaScript内置对象 所谓的内置对象 指的是:JavaScript本身就自己有的对象 可以直接拿来就用.例如Array String 等等.JavaScript一共有12内置对象 ...
- Spirit带你彻底搞懂JS的6种继承方案
JavaScript中实现继承的6种方案 01-原型链的继承方案 function Person(){ this.name="czx"; } function Student(){ ...
- 通过一张简单的图,让你彻底地搞懂JS的==运算
大家知道,JavaScript中的==是一种比较复杂运算,它的运算规则很奇怪,很容易让人犯错,从而成为JavaScript中“最糟糕的特性”之一. 在仔细阅读ECMAScript规范的基础上,我画了一 ...
随机推荐
- TortoiseGit拉取或推送,输入账号密码后提示 HTTP Basic: Access denied fatal: Authentication failed 解决方案
TortoiseGit拉取或推送项目,输入账号密码后,提示 HTTP Basic: Access denied fatal: Authentication failed. 大体意思是,HTTP基本认证 ...
- workbench使用小笔记(不定期持续更新)
1. 删除不使用的工作空间 在使用workbench时,之前可能建了好几个工作空间,现在有一些不使用了,每次打开都能还能看到它们,对于强迫症来说多少有一些不爽.如下图: 现在,就把那些不使用的工作空间 ...
- 关于pandas 调用mongodb出Memory error错误
其实就是内存不够了,我加载的数据是800多万,数据量太大出错下面试错误提示 C:\Users\souwayai\Miniconda3\python.exe D:/www/use_pandas/pd7. ...
- ElasticSearch(十一)Elasticsearch清空指定Index/Type数据
POST /index_name/type_name/_delete_by_query?conflicts=proceed { "query": { "match_all ...
- 【MySQL】redo log --- 刷入磁盘过程
1.redo log基本概念 redo log的相关概念这里就不再过多阐述,网上有非常多的好的资料,可以看下缥缈大神的文章:https://www.cnblogs.com/cuisi/p/652507 ...
- HTML与CSS的一些知识(二)
续: 5.表单标签<form></form> 用于收集用户信息,统一提交到服务器 一般用input标签收集,再用提交按钮提交:input标签根据type属性值不同有不同的类型: ...
- windows 10, v1903 正式版下载
一.简体中文版 cn_windows_10_business_editions_version_1903_x64_dvd_e001dd2c.iso sha1:bc6176bee6130 ...
- 1px解决方案--集锦
没有废话,直接上代码 汇聚各种版本,持续更新中.... 1.sass @charset "utf-8"; /** * @module 背景与边框 * @description 为元 ...
- 20175317 《Java程序设计》第八周学习总结
20175317 <Java程序设计>第八周学习总结 教材学习内容总结 第八周我学习了教材第十五章的内容,认识了什么是泛型与集合框架,具体内容如下: 泛型 1. 如何声明泛型类 2. 如何 ...
- 序列比对和构建进化树(clustalw和phylip)
安装clustalw很简单,不提了. 找了几个蛋白序列进行比对,命名为dm.fasta 1.输入 ./clustalw2 进入交互模式 2.选择1 并输入文件名字 3.输入2, 进行多序列比对 4. ...