不会JS中的OOP,你也太菜了吧!(第二篇)
一、你必须知道的
1> 原型及原型链在继承中起到了关键的作用。所以你一定要理解他们。2> 不会JS中的OOP,你也太菜了吧!(第一篇)
二、继承的6种方法
1> 原型链继承
原型链继承是通过创建Super的实例,并将该实例赋值给Sub.prototype来实现的。
实现的本质是:重写子类型的原型对象,代之以超类型的实例。
function Super(){
this.name = 'JChen___';
}
Super.prototype.getSuperName = function(){
return this.name;
}
function Sub(){
this.subname = 'JChen___son';
}
Sub.prototype = new Super(); //原型继承体现在这里
Sub.prototype.getSubName = function(){
return this.subname;
}
var instance = new Sub();
注意:此时instance.constructor现在指向的是Super的,这是因为Sub.prototype指向了Super.prototype,而Super.prototype.constructor = Super。
原型链的问题:类似于利用原型创建对象,原型共享的特性也是原型链继承的最大问题。
2> 借用构造函数继承
在解决原型中包含引用类型值所带来的问题的过程中,我们开始使用一种叫做借用构造函数的技术。
这种技术的基本思想相当简单:在子类型构造函数的内部调用超类型构造函数。
这样一来,就会在新子类对象上执行超类函数中定义的所有对象初始化代码。结果,每个子类的实力都会有自己的超类中属性的副本了。
function Super2(name){
this.colors = ['red', 'blue'];
this.name = name;
}
function Sub2(){
Super2.call(this, 'JChen___2'); //借用构造函数技术体现在这里
this.age = 29;
}
var instance1 = new Sub2();
instance1.colors.push('black');
var instance2 = new Sub2();
instance2.colors.push('green');
借助构造函数继承的问题:
1): 方法都在构造函数中定义,无法复用。
2): 在超类型的原型中的方法对子类是不可见的。
3> 组合继承(原型+借用构造)
组合继承指的是将原型链和借用构造函数的技术组合到一块,从而发挥二者之长的一种继承模式。
组合继承的思路:使用原型链实现对方法和属性的继承,通过借用构造函数实现对实例属性的继承。
function Super3(name){
this.name = name;
this.colors = ['red', 'blue'];
}
Super3.prototype.sayName = function(){
return this.name;
}
function Sub3(name, age) {
Super3.call(this, name);
this.age = age;
}
Sub3.prototype = new Super3(); //解决借用构造函数技术的缺点
Sub3.prototype.constructor = Sub3; //纠正原型继承改变了的构造函数
Sub3.prototype.sayAge = function(){
return this.age;
}
组合继承避免了原型链和借用构造函数的缺陷,融合了他们的优点,成为JavaScript中最常用的继承模式。
组合继承的问题:两次调用超类构造函数。
4> 原型式继承
原型式继承的思路:借助原型可以基于已有的对象创建新对象,同时还不必因此创建自定义类型。
function object(o){ //原型式继承的关键
function F(){}
F.prototype = o;
return new F();
}
var person = {
name: 'JChen___4',
colors: ['blue']
}
var person1 = object(person);
person1.name = 'JChen___4___2'
person1.colors.push('red');
var person2 = object(person);
person2.name = 'JChen___4___3';
person2.colors.push('green');
原型式继承的问题:同原型链一样,他也有共享的劣势。
5> 寄生式继承
寄生式继承的思路:创建一个仅用于封装继承过程的函数,该函数内部以某种方式来增强对象,最后再返回该对象
function createAnother(origin){ //寄生式继承的关键
var clone = object(origin);
clone.sayHi = function(){
return 'Hi';
};
return clone;
}
var person = {
name: 'JChen___4',
colors: ['blue']
}
var person1 = createAnother(person);
寄生式继承的问题:像构造函数一样,由于不能做到函数的复用而降低效率。
6> 寄生组合式继承
寄生组合式继承:通过借用构造函数来借用属性,通过原型链的混成形式来继承方法。
其背后的思想是:不必为了指定子类型的原型而调用超类型的构造函数,我们需要的无非就是超类型的一个副本而已。
function object(o){
function F(){}
F.prototype = o;
return new F();
}
function inheritProto(subType, superType){ //避免第一调用构造函数的关键
var proto = object(superType.prototype);
proto.constructor = subType;
subType.prototype = proto;
}
function Super6(name){
this.name = name;
this.colors = ['red', 'blue'];
}
Super6.prototype.sayName = function(){
return this.name;
}
function Sub6(name, age){
Super6.call(this, name);
this.age = age;
}
inheritProto(Sub6, Super6);
Sub6.prototype.sayAge = function(){
return this.age;
}
var instance1 = new Sub6('JChen___6', '12');
instance1.colors.push('black');
开发人员普遍认为寄生组合式继承是引用类型最理想的继承范式。
三、总结
这就是JavaScript中的6种继承方式,如果大家能够画出每个继承的原型链关系图,那么继承就是小菜一碟了。
不会JS中的OOP,你也太菜了吧!(第二篇)的更多相关文章
- 小结JS中的OOP(下)
关于JS中OOP的具体实现,许多大神级的JS专家都给出了自己的方案. 一:Douglas Crockford 1.1 Douglas Crockford实现的类继承 /** * 原文地址:http:/ ...
- 小结JS中的OOP(中)
此篇文章主要是提炼<JavaScript高级程序设计>中第六章的一些内容. 一:JS中OOP相关的概念 开始之前先总结JS中OOP相关的一些概念: 构造函数:JS中的构造函数就是普通的函数 ...
- 小结JS中的OOP(上)
前言:大家都知道,OOP有三大特性:封装,继承,多态.下面是自己对这三个特性的理解: 封装:把属性与方法整合到某种数据类型中.目的是让类的使用者按类的编写者的意愿去使用类.在封装过程中会一般会做两件事 ...
- js面向对象编程:this究竟代表什么?第二篇
总觉得自己弄明确了js中this的含义.this总是指向调用方法的对象,作为方法调用,那么this就是指实例化的对象.但前几天自己写脚本却遇到了一个非常奇怪的问题. 代码例如以下: ...
- 不会JS中的OOP,你也太菜了吧!(第一篇)
一.你必须知道的 1) 字面量 2) 原型 3) 原型链 4) 构造函数 5) 稳妥对象(没有公共属性,而且其方法也不引用this的对象.稳妥对象适合用在安全的环境中和防止数据被其它程序改变的时候) ...
- JavaScript面向对象(一)——JS OOP基础与JS 中This指向详解
前 言 JRedu 学过程序语言的都知道,我们的程序语言进化是从"面向机器".到"面向过程".再到"面向对象"一步步的发展而来.类似于 ...
- Cookie和Session在Node.JS中的实践(一)
Cookie和Session在Node.JS中的实践(一) Cookie和Session是一个非常有趣的概念,也是一个老生常谈的话题.然而,作者看了许多文章,也翻看了几本书籍,它们对Cookie和Se ...
- 进阶学习js中的执行上下文
在js中的执行上下文,菜鸟入门基础 这篇文章中我们简单的讲解了js中的上下文,今天我们就更进一步的讲解js中的执行上下文. 1.当遇到变量名和函数名相同的问题. var a = 10; functio ...
- JS中三目运算符和if else的区别分析与示例
本文是通过示例详细分析了JS中三目运算符和if else的区别,是篇非常不错的文章,这里推荐给大家. 今天写了一个图片轮播的小demo,用到了判断 先试了一下if else,代码如下: 复制代码代 ...
随机推荐
- SQL笔记-第二章,数据表的创建和管理
数据类型 分5类:整数.数值.字符相关.日期时间以及二进制 1.整数 数据库系统 类型 说明 MYSQL tinyint [unsigned] 一个很小的整数.有符号的范围是-128 到127,无符号 ...
- BF算法和KMP算法(javascript版本)
var str="abcbababcbababcbababcabcbaba";//主串 var ts="bcabcbaba";//子串 function BF( ...
- Python在金融,数据分析,和人工智能中的应用
Python在金融,数据分析,和人工智能中的应用 Python最近取得这样的成功,而且未来似乎还会继续下去,这有许多原因.其中包括它的语法.Python开发人员可用的科学生态系统和数据分析库.易于 ...
- ASP.NET的错误处理机制之一(概念)
对Web应用程序来说,发生不可预知的错误和异常在所难免,我们必须为Web程序提供错误处理机制.当错误发生时,我们必须做好两件事情:一是将错误信息记录日志,发邮件通知网站维护人员,方便技术人员对错误进行 ...
- php小算法总结一(数组重排,进制转换)
1.两个有序数组组合成一个新的有序数组 <?php $arr1=array(2,5,7,9,12); $arr2=array(3,4,6,8,10,11); function merge_sor ...
- C++读入一个参数
题目内容:已知正方形的边长,试编程求出其面积. 输入描述:输入不超过50个正整数的数据n(1<=n<=10000),每个正整数间以空格隔开. 输出描述:每次读入一个正整数,便输出其正方形的 ...
- Android开发之Source无法覆写public void onClick(View v)
初学Android开发,在为一个按钮[该按钮继承OnClickListener()]写监听时,发现无法在Source中引入public void onClick(View v),当时非常纳闷,平常情况 ...
- Tomcat工作原理
http://www.cnblogs.com/shootercheng/p/5838645.html
- MarkDown写作技巧
前言 年轻的我们往往苦恼于没有充实的社会经历.社会阅历,很难有较强的个人能力.个人魅力!就个人能力而言,本 人主要把它分为两种:“吸收能力”和“输出能力”.吸收能力主要体现了个人对外界知识的摄入能力, ...
- 小课堂week14 Google软件测试之道
读<Google软件测试之道> 在IT领域,Google是一面旗帜,是一家非常善于思考善于尝试的公司.随着面临挑战的不断增大,传统的测试开展方式也越来越力不从心,这本书讲述的就是一次完整的 ...