JavaScript - 动态加载类并使用特定属性创建新实例

问题描述 投票:0回答:1

我有一个Node.js v11.11.0应用程序。在这个应用程序中,我的文件结构如下:

./src
  /animals/
    animal.js
    tiger.js
    koala.js
  index.js

如上所示,我在animals目录中定义了三个类。随着时间的推移,我打算用更复杂的逻辑添加额外的动物。这时,我的类定义如下:

animal.js

'use strict';

class Animal {
  constructor(properties) {
    properties = properties || {};

    this.kind = 'Unknown';
  }

  eat(foods) {
    for (let i=0; i<foods.length; i++) {
      console.log(`Eating ${foods[i]}`);
    }
  }
}

module.exports = Animal;

tiger.js

'use strict';

const Animal = require('./animal');

class Tiger extends Animal {
  constructor(properties) {
    super(properties);

    this.kind = 'Tiger';
  }

  eat(foods) {
    for (let i=0; i<foods.length; i++) {
      if (foods[i].kind === 'meat') {
        console.log(`Eating ${foods[i]}`);
      }
    }
  }
}

module.exports = Tiger;

koala.js

'use strict';

const Animal = require('./animal');

class Koala extends Animal {
  constructor(properties) {
    super(properties);

    this.kind = 'Koala';
  }

  eat(foods) {
    for (let i=0; i<foods.length; i++) {
      if (foods[i].kind === 'plant') {
        console.log(`Eating ${foods[i]}`);
      }
    }
  }
}

module.exports = Koala;

我试图根据用户输入的内容动态创建其中一个类的实例。例如,我这样做:

index.js

const readline = require('readline').createInterface({
  input: process.stdin,
  output: process.stdout
})

readline.question('What animal file do you want to load?', (fname) => {
  let properties = {
    name: 'My Pet'
  };

  let Animal = require(fname);
  let pet = Object.create(Animal.prototype, properties);
});

如果我运行它并输入./animals/tiger.js,我会收到错误。错误说:Property description must be an object。我不明白这一点。 TigerKoala都延伸Animal。同时,由于我的列表将是动态的,我需要自由格式选项,键入文件路径并动态加载类。出于这个原因,我不能使用here所示的方法,它专门列出了类名。

如何动态加载类并使用特定属性创建该类的实例?

javascript es6-class
1个回答
1
投票

Object.create的第二个参数与你传递给Object.defineProperties的论点相同,因此你的properties无效。而是使用Object.assign:

 Object.assign(Object.create(Animal.prototype), properties)

但是你为什么不只是调用构造函数?

 new Animal(properties)

其次:

 properties = properties || {};

可能是:

 this.properties = properties || {};

我不建议动态require,尤其是不是来自用户输入的值,而是创建一个查找对象:

 const AnimalByName = {
   Bear: require("./Bear"),
   //...
 };

 const instance = new AnimalByName[name](properties);
© www.soinside.com 2019 - 2024. All rights reserved.