Organize prototype javascript while perserving object reference and inheritance

前端 未结 3 2105
小蘑菇
小蘑菇 2020-11-22 05:50

I have built a large application using JavaScript prototype and inheritance. But I am having a hard time organizing my code. For example I have a class carousel which has ma

相关标签:
3条回答
  • 2020-11-22 06:21

    You could make Controls a class of it's own:

    var Controls = function (controllable_object) {
        this.ref = controllable_object;
    };
    Controls.prototype.next = function () {
        this.ref.foo();
    }
    // ..
    
    var Carousel = function () {
        this.controls = new Controls(this);
    };
    // ..
    

    This doesn't allow you to override the implementation of Controls though. With more dependency injection you'd get something like:

    var Controls = function (controllable_object) {
        this.ref = controllable_object;
    };
    Controls.prototype.next = function () {
        this.ref.foo();
    }
    // ..
    
    var Carousel = function () {
            this.controllers = [];
        };
    Carousel.prototype.addController = function (controller) {
            this.controllers.push(controller);
        };
    // ..
    
    var carousel = new Carousel();
    carousel.addController(new Controls(carousel));
    
    0 讨论(0)
  • 2020-11-22 06:23

    You can use the .bind method of Function.

    In Javascript Functions inherit from Object, so they have their own methods. One of those methods is .bind:

    https://developer.mozilla.org/en-US/docs/JavaScript/Reference/Global_Objects/Function/bind

    Also you are doing inheritance wrong, the right way with raw Javascript is:

    ChildClass= function() {
        ParentClass.apply(this, arguments); //calling parent constructor
        //constructor
    };
    
    ChildClass.prototype= new ParentClass();
    

    Then you can simply do this on your constructor:

    Courossel= function() {
        ParentClass.apply(this, arguments); //calling parent constructor
        this.controls.next.bind(this);
        this.controls.prev.bind(this);
        this.controls.bindControls.bind(this);
    }
    

    But I have to say that Frits suggestion is better, make the controls their own class and instantiate it on Carousel constructor passing a reference to your Carousel instance (the this keyword). Just don't call it ".ref", it's confusing.

    0 讨论(0)
  • 2020-11-22 06:36

    My inheritance is done like this:

    $.extend(this.prototype , new parentClass);
    

    Ouch. This is not inheritance (with new BigCarousel instanceof Carousel), but just copying properties. Maybe this is enough for you, but then you should call it mixin. Also, you should avoid using new for inheritance.


    But this will cause the value of "this" being lost. How can I work around that?

    It's impossible to have this point to the parent object with nested properties (as long as you don't want to explicitly set it every time). You have only two choices:

    • Forget it, and organize your methods by prefixing them (controlNext, controlBind, …)
    • Give each of your carousels its own controls object. For inheritance, make them CarouselControls instances for example. This especially fits well if those controls are quite independent from the carousel, and don't need to access the carousel they're attached to everywhere. If they are not, you still can pass a reference to the parent carousel into their constructor for example:

      this.controls = new CarouselControls(this);
      

      Also, for customizing the controls in different carousels, you might have to subclass the CarouselControls as well - or you prepare your Controls object to serve for different carousels in general, so that from BigCarousel you can

      Carousel.call(this); // make this a carousel
      this.controls.activate({big: true, fast: false}); // or something
      
    0 讨论(0)
提交回复
热议问题