Node.js,需要文件夹中的所有模块并直接使用加载的模块

Posted

技术标签:

【中文标题】Node.js,需要文件夹中的所有模块并直接使用加载的模块【英文标题】:Node.js, require all modules in folder and use loaded module directly 【发布时间】:2015-10-17 21:20:47 【问题描述】:

MyModule 文件夹中,我有这两个 JS 文件。

SayHello.js

module.exports.SayHello = function() 
    return('Hello !');

SayByeBye.js

module.exports.SayByeBye = function() 
    return('Bye Bye!');

在 Node.js 中,我想要求 MyModule 文件夹中的所有文件并直接调用函数 SayHelloSayByeBye 类似:

require(./MyModule) 
console.log(SayHello());
console.log(SayByeBye());

编辑:

回答@Yanick Rochon,我这样做:

> ./app/my-module/index.js

global.SayHello = require('./my-module/SayHello').SayHello;
global.SayByeBye = require('./my-module/SayByeBye').SayByeBye;

> ./app/my-module/say-hello.js

module.exports.SayHello = function() 
    return('Hello !');
;

> ./app/my-module/say-byebye.js

module.exports.SayByeBye = function() 
    return('Bye Bye !');
;

> ./app/main.js

require('./my-module');
console.log(SayHello());
console.log(SayByeBye());

在节点文档中有一个关于global objects 的部分。

但是,应谨慎使用全局变量。通过向全局空间添加模块,我降低了可测试性和封装性。但在这种情况下,我认为使用这种方法是可以接受的。

【问题讨论】:

你不应该把模块放在导出之前,而是简单地做exports.SayHello = function.....看这里***.com/questions/31657341/… 丹尼尔,链接是这篇文章吗? @LeMoussel 哎呀***.com/questions/7137397/… 不相关,但为什么你的文件名是颠倒的?它比我愿意承认的更困扰我。 肯定有误。我更正了。 【参考方案1】:

它并不完美,但这样的事情应该可以帮助你完成这个:

var fs = require('fs');
var path = require('path');

var files = fs.readdirSync(__dirname);
var ownFilename = __filename.substr(__filename.lastIndexOf(path.delimiter) + 1);

var modules = ;
for (var i = 0; i < files.length; i++) 
        var filename = files[i];
        if (filename.substr(-3) === '.js' && filename !== ownFilename) 
                modules[filename.slice(0, -3)] = require('./' + filename);
        


console.log(modules.SayByeBye());
console.log(modules.SayHello());

【讨论】:

【参考方案2】:

第一件事...

我相信您将 Node.js 与 php 或 .Net 混淆了,因为您没有将在其他模块中导出的内容“导入”到当前模块中。除非您手动执行此操作,否则不会。例如,当您调用时

require('./my-module');

(请注意,我将您的 MyModule 重命名为 Node.js 命名约定。)

您不会将内容加载到当前上下文中;您只需加载脚本而不将其分配给任何内容。要访问 my-module 公开的内容,您需要分配它,或者直接使用它。例如:

require('./my-module').someFunction();

var myModule = require('./my-module');

myModule.someFunction();

模块不是命名空间,而是在其自身上下文之外公开公共属性的 javascript 对象(即使用 module.exports = ...

回答

你有两种最流行的方法来完成这个:

解决方案 1

在您要加载所有脚本的文件夹中创建一个index.json 文件。返回的 JSON 对象应该是自动加载的所有模块:

> ./app/index.json

[
  "say-hello.js",
  "say-goodbye.js"
]

您还应该考虑让所有文件 API 兼容:

> ./app/say-hello.js

module.exports = function sayHello() 
  return 'Hello !';
;

> ./app/say-goodbye.js

module.exports.sayGoodbye = function () 
  return 'Goodbye !';
;

然后像这样加载并执行所有内容:

var path = require('path');
var basePath = './app/';

var files = require(basePath);

var mods = files.forEach(function (loaded, file) 
  var mod = require(path.join(basePath, file));

  // mod is a function with a name, so use it!
  if (mod instanceof Function) 
    loaded[mod.name] = mod;
   else 
    Object.keys(mod).forEach(function (property) 
      loaded[property] = mod.property;
    );
  
, );

mods.sayHello();
mods.sayGoodbye();

解决方案 2

读取文件夹中的所有.js 文件并导入它们。我强烈建议您为此使用glob

var glob = require("glob")
var path = require('path');
var basePath = './app/';

var mods = glob.sync(path.join(basePath, '*.js')).reduce(function (loaded, file) 
  var mod = require(file);

  // mod is a function with a name, so use it!
  if (mod instanceof Function) 
    loaded[mod.name] = mod;
   else 
    Object.keys(mod).forEach(function (property) 
      loaded[property] = mod.property;
    );
  

  return loaded;
, );

mods.sayHello();
mods.sayGoodbye();

注意module.exportsexports的区别

通常使用module.exports === exports,但建议使用module.exports,原因如下

exports = function Foo()           // will not do anything
module.exports = function Foo()    // but this will do what you expect

// however these two lines produce the same result
exports.foo = 'Bar';
module.exports.foo = 'Bar';  

因此,在所有情况下都建议使用module.exports

【讨论】:

以上是关于Node.js,需要文件夹中的所有模块并直接使用加载的模块的主要内容,如果未能解决你的问题,请参考以下文章

使用 Node.js 在内存中缓冲整个文件

模块化方式构建Node.js应用程序

深入浅出Node.js 模块机制

Node.js 的 Global全局对象

Node.js中模块加载机制

Node.js中模块加载机制