util.inherits - alternative or workaround

后端 未结 4 769
面向向阳花
面向向阳花 2021-02-05 08:46

I am a n00b in node, and find util.inherits() very useful, except for the fact that it seems to replace the entire prototype of the original object. For instance:

相关标签:
4条回答
  • 2021-02-05 09:08

    It does not make sense that you have to declare your prototype after util.inherits(). My guess is util.inherits originated as an internal-use-only method, tailored only for the limited internal use-cases it was initially intended for, which at some point got published for general usage. The util module is written in pure JS, so it is very easy to implement your own version of util.inherit that preserves your prototype. Here's the original util.inherit source:

    exports.inherits = function(ctor, superCtor) {
      ctor.super_ = superCtor;
      ctor.prototype = Object.create(superCtor.prototype, {
        constructor: {
          value: ctor,
          enumerable: false,
          writable: true,
          configurable: true
        }
      });
    };
    

    As for the multiple inheritance, that's going to be a much more difficult problem to tackle, as Javascript's prototype inheritance is not really suited for multiple inheritance at all. Each instance has only a single internal [[Prototype]] property, which is used to look up members that are not found in the actual instance. You could merge the prototypes of two separate "parent classes" into a single prototype, but you will then lose the inheritance to their parents, and you will lose the ability to change the parent prototype and have all children see the change.

    0 讨论(0)
  • 2021-02-05 09:15

    You should first inherit, after the function definition, and then implement your object. Also don't forget to call the superclass constructor in your class's constructor.

    Function A(y){
      this.y = x;
    }
    
    Function B(x,y){
      this.x = x;
      A.call(this,y);
    }
    
    util.inherits(B,A);
    
    B.prototype.mythodA = function() {
        //do something
    }
    
    0 讨论(0)
  • 2021-02-05 09:19

    I very much wanted the same thing and was unhappy with extend, so I created this extension to the already useful util.inherits method:

    var util = require('util');
    
    module.exports = {
        inherits : function(sub, sup, proto) {
            util.inherits(sub, sup);
            if (typeof proto !== 'undefined') {
                Object.keys(proto).forEach(function(key) {
                    sub.prototype[key] = proto[key];
                });
            }
        }
    };
    

    I put this in my project's ./util/index.js and then do this to use it:

    var EventEmitter = require('events').EventEmitter;
    var util = require('./util');
    
    function Foo() {
        EventEmitter.call(this);
    }
    
    util.inherits(Foo, EventEmitter, {
        bar : function(){
            console.log(this instanceof EventEmitter); // true
        }
    });
    

    Maybe I'll publish this if I find it's robust and useful more and more. I just barely implemented it myself, so I'm giving it a test run.

    Let me know what you think!

    NOTE: This does override methods on either of the classes with the ones defined in the proto hash at the end. Just be aware of that.

    0 讨论(0)
  • 2021-02-05 09:20

    As of node version 5.0.0, util.inherits has been changed to support the behaviour you are looking for using the setPrototypeOf method:

    FirstBase.js

    function FirstBase(firstBaseProp){
        this.firstBaseProp = firstBaseProp;
    }
    
    FirstBase.prototype.getFirstBaseProp = function(){
        return this.firstBaseProp;
    };
    
    
    module.exports = FirstBase;
    

    SecondBase.js

    var FirstBase = require('./FirstBase.js'),
        util = require('util');
    
    function SecondBase(firstBaseProp, secondBaseProp){
        this.secondBaseProp = secondBaseProp;
        SecondBase.super_.apply(this, arguments);
    }
    
    SecondBase.prototype.getSecondBaseProp = function(){
        return this.secondBaseProp;
    };
    
    util.inherits(SecondBase, FirstBase);
    
    module.exports = SecondBase;
    

    ThirdBase.js

    var SecondBase = require('./SecondBase.js'),
        util = require('util');
    
    function ThirdBase(firstBaseProp, secondBaseProp, thirdBaseProp){
        this.thirdBaseProp = thirdBaseProp;
        ThirdBase.super_.apply(this, arguments);
    }
    
    ThirdBase.prototype.getThirdBase = function(){
        return this.thirdBaseProp;
    };
    
    util.inherits(ThirdBase, SecondBase);
    
    module.exports = ThirdBase;
    

    instance.js

    var ThirdBase = require('./ThirdBase.js');
    
    var instance = new ThirdBase('first', 'second', 'third');
    
    // With node < 5.0.0 (Object.create)
    console.log(instance.getFirstBaseProp()); // first
    console.log(instance.getSecondBaseProp()); // undefined
    console.log(instance.getThirdBase()); // undefined
    
    // With node >= 5.0.0 (Object.setPrototypeOf)
    console.log(instance.getFirstBaseProp()); // first
    console.log(instance.getSecondBaseProp()); // second
    console.log(instance.getThirdBase()); // third
    

    If you're running an older version of node that supports setPrototypeOf (0.12.x does), you can just export util.inherits and use it as an internal utility function.

    0 讨论(0)
提交回复
热议问题