JavaScript中的继承方式详解


Posted in Javascript onFebruary 11, 2015

js继承的概念

js里常用的如下两种继承方式:

原型链继承(对象间的继承)
类式继承(构造函数间的继承)
由于js不像java那样是真正面向对象的语言,js是基于对象的,它没有类的概念。所以,要想实现继承,可以用js的原型prototype机制或者用apply和call方法去实现

在面向对象的语言中,我们使用类来创建一个自定义对象。然而js中所有事物都是对象,那么用什么办法来创建自定义对象呢?这就需要用到js的原型:

我们可以简单的把prototype看做是一个模版,新创建的自定义对象都是这个模版(prototype)的一个拷贝 (实际上不是拷贝而是链接,只不过这种链接是不可见,新实例化的对象内部有一个看不见的__Proto__指针,指向原型对象)。

js可以通过构造函数和原型的方式模拟实现类的功能。 另外,js类式继承的实现也是依靠原型链来实现的。

原型式继承与类式继承

类式继承是在子类型构造函数的内部调用超类型的构造函数。
严格的类式继承并不是很常见,一般都是组合着用:

function Super(){

    this.colors=["red","blue"];

}
function Sub(){

    Super.call(this);

}

原型式继承是借助已有的对象创建新的对象,将子类的原型指向父类,就相当于加入了父类这条原型链

原型链继承

为了让子类继承父类的属性(也包括方法),首先需要定义一个构造函数。然后,将父类的新实例赋值给构造函数的原型。代码如下:

<script>

    function Parent(){

        this.name = 'mike';

    }
    function Child(){

        this.age = 12;

    }

    Child.prototype = new Parent();//Child继承Parent,通过原型,形成链条
    var test = new Child();

    alert(test.age);

    alert(test.name);//得到被继承的属性

    //继续原型链继承

    function Brother(){   //brother构造

        this.weight = 60;

    }

    Brother.prototype = new Child();//继续原型链继承

    var brother = new Brother();

    alert(brother.name);//继承了Parent和Child,弹出mike

    alert(brother.age);//弹出12

</script>

以上原型链继承还缺少一环,那就是Object,所有的构造函数都继承自Object。而继承Object是自动完成的,并不需要我们自己手动继承,那么他们的从属关系是怎样的呢?

确定原型和实例的关系

可以通过两种方式来确定原型和实例之间的关系。操作符instanceof和isPrototypeof()方法:

alert(brother instanceof Object)//true

alert(test instanceof Brother);//false,test 是brother的超类

alert(brother instanceof Child);//true

alert(brother instanceof Parent);//true

只要是原型链中出现过的原型,都可以说是该原型链派生的实例的原型,因此,isPrototypeof()方法也会返回true

在js中,被继承的函数称为超类型(父类,基类也行),继承的函数称为子类型(子类,派生类)。使用原型继承主要由两个问题:
一是字面量重写原型会中断关系,使用引用类型的原型,并且子类型还无法给超类型传递参数。

伪类解决引用共享和超类型无法传参的问题,我们可以采用“借用构造函数”技术

借用构造函数(类式继承)

<script>

    function Parent(age){

        this.name = ['mike','jack','smith'];

        this.age = age;

    }
    function Child(age){

        Parent.call(this,age);

    }

    var test = new Child(21);

    alert(test.age);//21

    alert(test.name);//mike,jack,smith

    test.name.push('bill');

    alert(test.name);//mike,jack,smith,bill

</script>

借用构造函数虽然解决了刚才两种问题,但没有原型,则复用无从谈起,所以我们需要原型链+借用构造函数的模式,这种模式称为组合继承

组合继承

<script>

    function Parent(age){

        this.name = ['mike','jack','smith'];

        this.age = age;

    }

    Parent.prototype.run = function () {

        return this.name  + ' are both' + this.age;

    };

    function Child(age){

        Parent.call(this,age);//对象冒充,给超类型传参

    }

    Child.prototype = new Parent();//原型链继承

    var test = new Child(21);//写new Parent(21)也行

    alert(test.run());//mike,jack,smith are both21

</script>

组合式继承是比较常用的一种继承方法,其背后的思路是 使用原型链实现对原型属性和方法的继承,而通过借用构造函数来实现对实例属性的继承。这样,既通过在原型上定义方法实现了函数复用,又保证每个实例都有它自己的属性。

call()的用法:调用一个对象的一个方法,以另一个对象替换当前对象。

call([thisObj[,arg1[, arg2[, [,.argN]]]]])

原型式继承

这种继承借助原型并基于已有的对象创建新对象,同时还不用创建自定义类型的方式称为原型式继承

<script>

     function obj(o){

         function F(){}

         F.prototype = o;

         return new F();

     }

    var box = {

        name : 'trigkit4',

        arr : ['brother','sister','baba']

    };

    var b1 = obj(box);

    alert(b1.name);//trigkit4
    b1.name = 'mike';

    alert(b1.name);//mike
    alert(b1.arr);//brother,sister,baba

    b1.arr.push('parents');

    alert(b1.arr);//brother,sister,baba,parents
    var b2 = obj(box);

    alert(b2.name);//trigkit4

    alert(b2.arr);//brother,sister,baba,parents

</script>

原型式继承首先在obj()函数内部创建一个临时性的构造函数 ,然后将传入的对象作为这个构造函数的原型,最后返回这个临时类型的一个新实例。

寄生式继承

这种继承方式是把原型式+工厂模式结合起来,目的是为了封装创建的过程。

<script>

    function create(o){

        var f= obj(o);

        f.run = function () {

            return this.arr;//同样,会共享引用

        };

        return f;

    }

</script>

组合式继承的小问题

组合式继承是js最常用的继承模式,但组合继承的超类型在使用过程中会被调用两次;一次是创建子类型的时候,另一次是在子类型构造函数的内部

<script>

    function Parent(name){

        this.name = name;

        this.arr = ['哥哥','妹妹','父母'];

    }
    Parent.prototype.run = function () {

        return this.name;

    };
    function Child(name,age){

        Parent.call(this,age);//第二次调用

        this.age = age;

    }
    Child.prototype = new Parent();//第一次调用

</script>

以上代码是之前的组合继承,那么寄生组合继承,解决了两次调用的问题。

寄生组合式继承

<script>

    function obj(o){

        function F(){}

        F.prototype = o;

        return new F();

    }

    function create(parent,test){

        var f = obj(parent.prototype);//创建对象

        f.constructor = test;//增强对象

    }
    function Parent(name){

        this.name = name;

        this.arr = ['brother','sister','parents'];

    }
    Parent.prototype.run = function () {

        return this.name;

    };
    function Child(name,age){

        Parent.call(this,name);

        this.age =age;

    }
    inheritPrototype(Parent,Child);//通过这里实现继承
    var test = new Child('trigkit4',21);

    test.arr.push('nephew');

    alert(test.arr);//

    alert(test.run());//只共享了方法
    var test2 = new Child('jack',22);

    alert(test2.arr);//引用问题解决

</script>

call和apply

全局函数apply和call可以用来改变函数中this的指向,如下:

// 定义一个全局函数

    function foo() {

        console.log(this.fruit);

    }
    // 定义一个全局变量

    var fruit = "apple";

    // 自定义一个对象

    var pack = {

        fruit: "orange"

    };
    // 等价于window.foo();

    foo.apply(window);  // "apple",此时this等于window

    // 此时foo中的this === pack

    foo.apply(pack);    // "orange"
Javascript 相关文章推荐
JQuery this 和 $(this) 的区别
Aug 23 Javascript
Javascript 八进制转义字符(8进制)
Apr 08 Javascript
JavaScript改变HTML元素的样式改变CSS及元素属性
Nov 12 Javascript
JS正则验证邮箱的格式详细介绍
Nov 19 Javascript
js实现倒计时时钟的示例代码
Dec 17 Javascript
使用javascript实现监控视频播放并打印日志
Jan 05 Javascript
javascript判断css3动画结束 css3动画结束的回调函数
Mar 10 Javascript
Bootstrap每天必学之级联下拉菜单
Mar 27 Javascript
AngularJS 模块详解及简单实例
Jul 28 Javascript
jQuery 全选 全不选 事件绑定的实现代码
Jan 23 Javascript
vue cli 3.0 搭建项目的图文教程
May 17 Javascript
Vue-cli3.x + axios 跨域方案踩坑指北
Jul 04 Javascript
JavaScript中原型和原型链详解
Feb 11 #Javascript
Node.js中的缓冲与流模块详细介绍
Feb 11 #Javascript
javascript中var的重要性分析
Feb 11 #Javascript
JavaScript设计模式之工厂模式和构造器模式
Feb 11 #Javascript
js实现可兼容IE、FF、Chrome、Opera及Safari的音乐播放器
Feb 11 #Javascript
45个JavaScript编程注意事项、技巧大全
Feb 11 #Javascript
JS实现判断碰撞的方法
Feb 11 #Javascript
You might like
php实现改变图片直接打开为下载的方法
2015/04/14 PHP
PHP检测用户语言的方法
2015/06/15 PHP
使用PHP处理数据库数据如何将数据返回客户端并显示当前状态
2016/02/16 PHP
PHPMAILER实现PHP发邮件功能
2018/04/18 PHP
PHP whois查询类定义与用法示例
2019/04/03 PHP
Laravel框架Request、Response及Session操作示例
2019/05/06 PHP
css动画效果之animation的常用样式
2021/03/09 HTML / CSS
Jquery+JSon 无刷新分页实现代码
2010/04/01 Javascript
JavaScript对象学习经验整理
2013/10/12 Javascript
在javascript中实现函数数组的方法
2013/12/25 Javascript
JavaScript实现复制内容到粘贴板代码
2016/03/31 Javascript
Node.js Streams文件读写操作详解
2016/07/04 Javascript
微信小程序 wx:key详细介绍
2016/10/28 Javascript
Bootstrap基本组件学习笔记之导航(10)
2016/12/07 Javascript
JavaScript取得gridview中获取checkbox选中的值
2017/07/24 Javascript
浅谈angularJs函数的使用方法(大小写转换,拷贝,扩充对象)
2018/10/08 Javascript
JS实现水平遍历和嵌套递归操作示例
2019/08/15 Javascript
使用Python获取并处理IP的类型及格式方法
2018/11/01 Python
Opencv实现抠图背景图替换功能
2019/05/21 Python
使用python 将图片复制到系统剪贴中
2019/12/13 Python
Numpy 多维数据数组的实现
2020/06/18 Python
python 制作python包,封装成可用模块教程
2020/07/13 Python
Django model重写save方法及update踩坑详解
2020/07/27 Python
python 利用panda 实现列联表(交叉表)
2021/02/06 Python
CSS去掉A标签(链接)虚线框的方法
2014/04/01 HTML / CSS
HTML5离线缓存在tomcat下部署可实现图片flash等离线浏览
2012/12/13 HTML / CSS
New Balance加拿大官方网站:运动鞋和健身服装
2018/11/19 全球购物
Club Monaco加拿大官网:设计师男女服装
2019/09/29 全球购物
戴尔新西兰官网:Dell New Zealand
2020/01/07 全球购物
size?荷兰官方网站:英国高级运动鞋精品店
2020/07/24 全球购物
报到证办理个人委托书
2014/10/06 职场文书
财务负责人岗位职责
2015/02/03 职场文书
公司员工管理制度
2015/08/04 职场文书
班委竞选稿范文
2015/11/21 职场文书
课文《燕子》教学反思
2016/02/17 职场文书
浅谈为什么我的 z-index 又不生效了
2022/07/15 HTML / CSS