zoukankan      html  css  js  c++  java
  • JS继承

      谈到面向对象编程就避不开继承这个概念,JS的继承主要依赖原型链来实现的,今天主要总结一下在JS中的多种继承方式。主要内容如下:

    1. 什么叫原型链
    2. 原型链继承
    3. 经典继承
    4. 组合继承
    5. 原型式继承
    6. 寄生组合式继承
    7. 简析JQuery的Extend函数
    8. 简析ExtJs的Extend函数

    1.什么是原型链

      原型与实例的链状结构叫做原型链。

      构造函数prototype指向一个对象的实例,利用这个构造函数创建实例的时候,实例对象就会拥有原型的属性和方法。如果这个构造函数的原型是另一个构造函数的实例,而另一个构造函数又是别的构造函数的实例,如此层层递进的关系构成实例与原型的链条,我们把这种链状结构叫做原型链。

      JS的继承主要通过原型链来实现的

    2.原型链继承

      原型链继承的结构如下:

     1 function Sup() {
     2     this.name = "sub...";
     3     this.getName = function() {
     4         return this.name;
     5     };
     6 }
     7         
     8 function Sub() {}
     9 Sub.prototype = new Sup();
    10         
    11 var obj = new Sub();
    12 alert(obj.getName());//sub...

      定义了一个超类Sup和一个子类Sub,子类的原型指向超类的实例,所以在通过子类构造的实例也拥有超类的name以及getName方法。当然,如果子类也定义了自己的name属性,调用getName函数返回的将是子类的name。

      构造函数构造函数的时候会自动添加一个构造器(constructor)指向当前的构造函数。所有以构造函数形式产生的原型实例,最后都会指向Objec对象,这也是它们拥有Objec定义的函数的原因。

    3.经典继承

      经典继承又叫借用构造函数,伪造对象等等。主要运用apply与call函数。

      apply与call除了参数有区别外,功能是一样的,都是改变调用者的上下文环境。这里只展示借用构造函数的示例,不详细探讨两个函数的具体细节。

      经典继承的结构如下:

     1 function Sup() {
     2     this.name = "sup...";
     3     this.getName = function() {
     4         return this.name;
     5     };
     6 }
     7 
     8 function Sub() {
     9     Sup.apply(this, arguments);
    10 }
    11 
    12 var obj = new Sub();
    13 alert(obj.getName());//sup...

      定义一个超类Sup和一个子类Sub,其中子类的代码是超类调用apply函数。Sup.apply(this, arguments)表示是在this的环境下执行Sup这个构造函数,效果等同于Sub也拥有了name与getName两个属性。

      需要强调的是它不会继承超类原型中的属性。即在Sup函数下面加一段Sup.prototype.getAge = function() {//...},Sub实例里是访问不到getAge的。

    4.组合继承

      组合继承是引用最广泛的继承方式,它弥补了原型链继承与经典继承的缺点,达到取长补短。

      原型链继承方式主要优点是共享,但这也恰恰成为了它的缺点。我们知道,在JS中对象的求职策略是按引用传递的,这意味着超类定义的对象将会被共享,任何一个用子类构造函数构造的实例修改了该对象,都会反映到其它子类实例中去(当然,超类中的非对象属性没有这种弊端)。

     1 function Sup() {
     2     this.arr = [1, 2, 3];
     3     this.getArr = function() {
     4         return this.arr.toString();
     5     };
     6 }
     7 
     8 function Sub() {}
     9 Sub.prototype = new Sup();
    10 
    11 var obj1 = new Sub();
    12 var obj2 = new Sub();
    13 obj1.arr.push(4);
    14 alert(obj2.getArr());//1,2,3,4

      经典继承不会有原型链的那种弊端,但却无法做倒我们原型设计的初衷——共享。在子类调用构造函数效果等同于在子类中把超类代码重新拷贝一遍,事实上,子类继承的超类函数是完全一样的。

      我们希望需要被共享的属性被共享,不需要被共享的属性不被共享,于是产生了原型链与借用构造函数的组合继承方式。

      组合继承方式如下:

    function Sup() {
        this.arr = [1, 2, 3];//继承后不需要共享的属性
    }
    Sup.prototype.getArr = function() {//借用构造函数不会继承超类原型里的东西
        return this.arr.toString();
    };
    
    function Sub() {
        Sup.apply(this);
    }
    Sub.prototype = new Sup();
    
    var obj1 = new Sub();
    var obj2 = new Sub();
    obj1.arr.push(4);
    alert(obj1.getArr());//1,2,3,4
    alert(obj2.getArr());//1,2,3

      组合继承足以应对各种继承场景了,但是道格拉斯.克罗克福德提出了一种基于已有的类型创建对象的继承方式——原型式继承。

    5.原型式继承

      原型式继承在ECMAJavascript 5中被定义到Object对象中的create中。

      原型式继承实现思路是定义一个创建函数,将原型对象作为参数,该函数内部封装了继承细节。

      其实现大致如下:

     1 function create(proto) {
     2     function F() {}
     3     F.prototype = proto;
     4     return new F();
     5 }
     6 
     7 var Book = {publish : "xx出版社"};
     8 
     9 var book = create(Book);
    10 alert(book.publish);//xx出版社

      EcmaJavascript5中只需把上面的create函数换成Object.create()即可,这个函数的第一个参数是原型对象,第二个参数是额外的属性对象用以控制属性对应内置属性。这里不做细节描述。

    6.寄生组合式继承

      寄生组合式继承是对组合式继承的优化,因为组合式继承会执行两次构造函数(创建子类原型时、借用构造函数时)。

      寄生组合式继承把继承原型的步骤放在原型式继承中进行。

      其代码结构如下:

    function inheritprototype(sub, sup) {
        var prototype = Object.create(sup.prototype);
        prototype.constructor = sub;//构造器
        sub.prototype = prototype;
    }
    
    function SupObject() {
        this.name = arguments[0] || "超类";
    }
    SupObject.prototype.getName = function() {
        return this.name;
    }
    
    function SubObject() {
        SupObject.apply(this, arguments);
    }
    inheritprototype(SubObject, SupObject);
    
    var s = new SubObject("子类");
    alert(s.getName());//子类

    7.JQuery中的Extend函数

      JQuery的继承有个特点:它有个可选参数(deep),用来标识执行浅复制或者深复制。

      用法:JQuery.extend([是否深度复制], 返回的目标对象, [扩展的对象...]);

      深复制与浅复制的区别在于前者会遍历同名属性的值,并区别复制。

      浅复制:扩展对象里的属性与目标对象的属性同名,则直接把拓展对象的属性值赋值给目标对象对应的属性。比如说目标对象是{a:{b:'b', c:'c'}},拓展对象是{a:{d:'d'}},执行浅复制则返回{a:{d:'d'}}。

      深复制:扩展对象里的属性与目标对象的属性同名,则遍历扩展对象(递归的过程),执行属性的复制。比如上面那个例子返回的结果是{a:{b:'b', c:'c',d:'d'}}。

    //浅复制
    function copy() {
        var target = arguments[0],
            config = arguments[1];
        for(var prop in config) {
            target[prop] = config[prop];
        }
        return target;
    }
    
    var target = {
        a : {
            b : 'b', 
            c : 'c'
        }
    }
    var config = {
        a : {
            d : 'd'
        }
    }
    
    var result = copy(target, config);//{a:{d:'d'}}
    //深复制
    function copy() {
        var target = arguments[0],
            config = arguments[1];
        for(var prop in config) {
            if(typeof target[prop] === 'object') {
                var temp;
                if(Object.prototype.toString.call(target[prop]) === '[object Array]') {
                    temp = target[prop] || [];
                }else {
                    temp = target[prop] || {};
                }
                target[prop] = copy(temp, config[prop]);
            }else {
                target[prop] = config[prop];
            }
        }
        return target
    }
    
    var target = {
        a : {
            b : 'b', 
            c : 'c'
        }
    }
    var config = {
        a : {
            d : 'd'
        }
    }
    
    var result = copy(target, config);//{a:{b:'b', c:'c',d:'d'}}

      在深复制代码里首先用typeof操作符把基本类型(string、boolean、number) 与引用类型区分开来,基本类型拷贝是覆盖(与浅复制相同),需要注意的是引用类型中的数组要给它声明一个空数组。

      在JQuery源码中代码复杂些,但功能也更强大。它增加了防循环引用机制、深复制时拒绝window、DOM、原型继承而来的复杂对象。

    8.ExtJs中的Extend

      ExtJs中的apply函数其实就是一种浅复制,而它的applyIf是一种忽略目标对象已有的属性。

      ExtJs的extend函数功能比较复杂,大致做了如下几种工作:

    • 确认子类型
    • 原型链继承
    • 给子类赋予覆盖函数(override)
    • 扩展对象对子类进行覆盖操作
    • 重写子类的继承函数

      1)确认子类型:Ext.extend函数有两种传参方式,三个参数的时候,从左到右传递的分别是子类、超类、扩展对象;两个参数的时候分别是超类与扩展对象。两种传参方式让人用起来很方便其内部实现大致如下:

    function confirSub(sub, sup, config) {
        if(!!sup && Object.prototype.toString.call(sup) === '[object Object]') {
            config = sup;
            sup = sub;
            if(sp.constructor != Object.prototype.constructor) {
                sub = sup.constructor;
            }else {
                sup.apply(this, arguments);
            }
        }
    }

      上面代码主要是处理两个参数情况,当第二个参数是对象字面量(扩展对象)时,把它当只传递两条参数处理。然后扩展对象、超类、子类进行重新定位,以便后文与传三个参数的情况统一。其中超类如果是新建对象,则子类用经典继承方式确认,如果超类是通过某些继承机制产生的对象,则将超类的构造器赋给子类。

      2)原型链继承:这个继承代码与寄生组合式继承原型中函数的代码很像。代码大致如下:

    function extendProto(sub, sup, config) {
        var F = function() {},
            sbp,//用来指代子类的原型
            spp = sup.prototype;//指代超类的原型
        F.prototyoe = spp;
        sbp = sub.prototype = new F();
        sbp.constructor = sub;
        //在控件子类继承中,扩展对象里的构造函数常调用它,
        //调用方式是:新类型.supperclass.constructor.call(this,arguments)
        sub.supperclass = spp;
    }

      3)子类新增覆盖函数(override),覆盖函数与JQuery浅复制类似,这里就不赘言了。

      4)扩展对象对子类进行浅复制:源码里就一行语句Ext.override(sb, overrides);

      5)重写子类extend函数:源码里也是一行解决sb.extend = function(o) {return Ext.extend(sb, o);};其作用是子类的子类只能把当前子类作为超类。

     

  • 相关阅读:
    android 入门-Service
    android 入门-Activity及 字体
    android 入门-安装环境
    PS 零基础训练1
    wp8 入门到精通 ImageCompress 图片压缩
    wp8 入门到精通 Gallery
    C# DateTime时间格式转换为Unix时间戳格式
    wp8 入门到精通 MultiMsgPrompt
    C# Func<T,TResult>
    redis简介
  • 原文地址:https://www.cnblogs.com/longhx/p/5404692.html
Copyright © 2011-2022 走看看