JavaScript中的原型链和继承
理解原型链
在 JavaScript 的世界中,函数是一等公民。
上面这句话在很多地方都看到过。用我自己的话来理解就是:函数既当爹又当妈。“当爹”是因为我们用函数去处理各种“粗活累活”(各种工具函数、页面交互、业务逻辑等);“当妈”是因为函数还会“生孩子”(创建对象)。
在 JavaScript 的世界中,每一个对象都有一个隐藏的__proto__
属性。这个属性指向生成这个对象的构造函数的原型(prototype
)。事实上,所有函数都有一个原型。考虑如下例子:
function Human() {}
var h1 = new Human()
var h2 = new Human()
// h1和h2都是是由Human函数创建出来的对象(称为Human的实例对象),所以:
h1.__proto__ = Human.prototype
h2.__proto__ = Human.prototype
// Human.prototype是一个对象(且称为原型对象吧),因此可以给它添加一些属性或者方法,比如:
Human.prototype.legs = 2
Human.prototype.sleep = function() {
console.log('sleeping....')
}
这边又规定了:
所有实例对象都可以访问到相应原型对象里面的属性和方法。(理解原型链的关键)
// 不难得出结论:
h1.legs === 2 // true
h2.legs === 2 // true
h1.sleep() // sleeping....
h2.sleep() // sleeping....
不要问我为什么能访问,我也不知道,因为下面这个人就是这么规定的。
实现继承
再看另外一个构造函数:
function Coder() {}
// 如果在这里把Human的原型赋给Coder的原型
Coder.prototype = Human.prototype
// 接着再添加一些Coder自己的原型方法和属性
Coder.prototype.age = 24
Coder.prototype.work = function() {
console.log('write code')
}
// 新建一个Coder对象
var c1 = new Coder()
// 不难得出
c1.age === Coder.prototype.age // true 24
c1.work === Code.prototype.work // true
c1.legs === Human.prototype.legs // true 2
c1.sleep === Human.prototype.sleep // true
虽然把 Human 的原型直接赋给 Coder 的原型可以让 Coder 的实例访问到 Human 原型上的属性和方法,但是如果想单独修改 Coder 原型的话会影响到 Human 原型上的值,这显然不是我们所期望的。
Coder.prototype.sleep = function() {
console.log('still write code....')
}
// 修改了Coder的原型,但同时也会影响Human的原型
// 因此需要修改Coder.prototype的指向从而避免这个问题
Coder.prototype = h1 // h1是Human的一个实例
所以应该这样写:
Coder.prototype = new Human()
// 或者
Coder.prototype = Object.create(Human.prototype)
// 实际上这两种写法还是有一定的区别
从而实现了 JavaScript 最简单的原型链式继承。
不过这么写还是会有以下两个缺陷:
父类构造函数中的引用类型的属性(方法)是所有子类实例共享的,所以会出现改动一个实例的属性(或方法)会影响所有实例的属性(或方法)。
创建子类实例时,无法向父类构造函数传参。
所以需要在子类构造函数内调用一下父类构造函数,并把子类的this
绑定上去:
function Coder(arg) {
Human.call(this, arg)
// 接着再写自身的属性(方法)
}
这就相当于把父类构造函数在子类构造函数中重写了一遍,也就不会出现【引用】问题了。上面就是构造函数继承。
参考
JavaScript中的原型链和继承的更多相关文章
- 图解JavaScript中的原型链
转自:http://www.jianshu.com/p/a81692ad5b5d typeof obj 和 obj instanceof Type 在JavaScript中,我们经常用typeof o ...
- 对Javascript 类、原型链、继承的理解
一.序言 和其他面向对象的语言(如Java)不同,Javascript语言对类的实现和继承的实现没有标准的定义,而是将这些交给了程序员,让程序员更加灵活地(当然刚开始也更加头疼)去定义类,实现继承 ...
- 一张图搞懂 Javascript 中的原型链、prototype、__proto__的关系 转载加自己的总结
1. JavaScript内置对象 所谓的内置对象 指的是:JavaScript本身就自己有的对象 可以直接拿来就用.例如Array String 等等.JavaScript一共有12内置对象 ...
- Javascript中的原型链、prototype、__proto__的关系
javascript 2016-10-06 1120 9 上图是本宝宝用Illustrator制作的可视化信息图,希望能帮你理清Javascript对象与__proto__.prototype和 ...
- JavaScript系列--浅析原型链与继承
一.前言 继承是面向对象(OOP)语言中的一个最为人津津乐道的概念.许多面对对象(OOP)语言都支持两种继承方式::接口继承 和 实现继承 . 接口继承只继承方法签名,而实现继承则继承实际的方法.由于 ...
- JavaScript中的原型链原理
工作中经常解除到prototype的概念,一开始错误的认为prototype是对象的原型链,其实prototype只能算是JavaScript开放出来的原型链接口,真正的原型链概念应该是__proto ...
- js 中的原型链与继承
ECMAScript中将原型链作为实现继承的主要方法,其基本思想是利用原型让一个引用类型继承另一个引用类型的属性和方法. 1.原型链 先回忆一下构造函数和原型以及实例的关系:每个构造函数都有一个原型对 ...
- JavaScript之基于原型链的继承
本文介绍下js的OOP中的继承. 上图的要点为:Foo函数在创建时会自动生成内置属性prototype,而typeof Foo.prototype是object类型的. 上图的要点为:Foo.prot ...
- Javascript中的原型链,__proto__和prototype等问题总结
1.js中除了原始数据类型 都是对象. 包括函数也是对象,可能类似于C++函数对象把 应该是通过解释器 进行()操作符重载或其他操作, 用的时候把它当函数用就行 但是实际上本质是一个对象 原型也是一个 ...
随机推荐
- odoo开发笔记 -- tree视图按指定字段group_by分组显示
注意点: view视图文件中的搜索视图要放在action视图的前边. 在action视图中,要写search_view_id, 还有context, search_default_group_by_字 ...
- 使用 IdentityServer4 实现 OAuth 2.0 与 OpenID Connect 服务
IdentityServer4 是 ASP.NET Core 的一个包含 OIDC 和 OAuth 2.0 协议的框架.最近的关注点在 ABP 上,默认 ABP 也集成 IdentityServer4 ...
- mysql 开发基础系列2 整型数据类型
Mysql 的数据类型 1. 对整数类型, Mysql 还支持类型名称后面的小括号内指定的显示宽度,例如int(5) 表示宽度小于5位时填满宽度,如果不显示指定宽度默认是int(11),一般配合zer ...
- Android UI(一)Layout 背景局部Shape圆角设计
Jeff Lee blog: http://www.cnblogs.com/Alandre/ (泥沙砖瓦浆木匠),retain the url when reproduced ! Thanks ...
- 07 训练Tensorflow识别手写数字
打开Python Shell,输入以下代码: import tensorflow as tf from tensorflow.examples.tutorials.mnist import input ...
- 跟面试官聊.NET垃圾收集,直刺面试官G点
装逼的面试官和装逼的程序员 我面试别人的时候,经常是按这种路子来面试: 看简历和面试题,从简历和面试题上找到一些技术点,然后跟应聘者聊. 聊某个技术点的时候,应聘者的回答会牵涉到其他的技术点,然后我会 ...
- 踏上编程大道 从 Python 开始成为神级 Coder
电脑科学,或说计算机科学,是个在美国不断成长的产业,薪资报酬也很高.市场上永远存在著对天赋异禀的新锐工程师的需求,这就是为什麽「学习程式语言」一直是一件有魅力的事情. 但是,就跟任何技能一样,我们常常 ...
- 注解@CrossOrigin解决跨域问题
注解@CrossOrigin 出于安全原因,浏览器禁止Ajax调用驻留在当前原点之外的资源.例如,当你在一个标签中检查你的银行账户时,你可以在另一个选项卡上拥有EVILL网站.来自EVILL的脚本不能 ...
- 【Java每日一题】20170227
20170224问题解析请点击今日问题下方的“[Java每日一题]20170227”查看(问题解析在公众号首发,公众号ID:weknow619) package Feb2017; import jav ...
- AnnotationConfigBeanDefinitionParser are only available on JDK 1.5 and higher
報錯: org.springframework.beans.factory.BeanDefinitionStoreException: Unexpected exception parsing XML ...