首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Node.js和模块作用域:将文件读入内存的最有效方法

Node.js和模块作用域:将文件读入内存的最有效方法
EN

Stack Overflow用户
提问于 2018-08-02 22:06:33
回答 1查看 201关注 0票数 2

我试图在变量实例化的上下文中更好地理解node.js模块的作用域和要求。更具体地说,将文件读入内存。

我有一个http服务器,它有一个模块,可以读取存储在代码库中的静态sql文件,并执行其中包含的查询。例如:

代码语言:javascript
复制
'use strict';

const fs = require('fs')
const executeSql = require('./utils/execute-sql');

module.exports.getDataById = (id) => {
  const sql = fs.readFileSync(
    `./data-access/sql/getDataById.sql`, 'utf8'
  );

  return executeSql(sql, id);
}

module.exports.getDataByName = (name) => {
  const sql = fs.readFileSync(
    `./data-access/sql/getDataByName.sql`, 'utf8'
  );

  return executeSql(sql, name);
}

我的理解是,每次调用这些函数(getDataByIdgetDataByName)时,都会以阻塞的方式同步读取文件,并阻塞执行线程。我知道我可以异步读取文件来避免这种情况,但我真正好奇的是,将sql变量从函数中提取到模块作用域中是否意味着readFile操作只发生一次(在实例化节点进程时),并且最终会更高效。例如:

代码语言:javascript
复制
'use strict';

const fs = require('fs')
const executeSql = require('./utils/execute-sql');
const sql1 = fs.readFileSync(
  `./data-access/sql/getDataById.sql`, 'utf8'
);
const sql2 = fs.readFileSync(
  `./data-access/sql/getDataByName.sql`, 'utf8'
);

module.exports.getDataById = (id) => {
  return executeSql(sql1, id);
}

module.exports.getDataByName = (name) => {
  return executeSql(sql2, name);
}

我知道require会在节点进程初始化时同步加载模块,并在其他地方需要这些模块时进一步缓存这些模块,但我想了解的是,如果不使用require的标准变量声明会导致类似的实例化内存引用,该引用在节点进程的生命周期中一直存在,而不需要在每次需要该模块时重新实例化。

我很感谢你能提供的任何见解。

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2018-08-02 22:14:26

你是正确的。每次模块需要另一个模块时,只有第一次执行代码时才会返回缓存的exports,因此在您的示例中,fs.readFileSync将运行一次(当有人第一次需要它时),node.js将缓存exports对象,并且在下一次中要求返回exports对象,而不会再次运行代码。

你可以用下面这样的代码来测试:

代码语言:javascript
复制
var mod = require("./myModule");
console.log(mod.nonExistantProperty); // This will log undefined
mod.nonExistantProperty = "yay";

var requireagain = require("./myModule");
console.log(requireagain.nonExistantProperty); // This will log yay

在第二个require中,不是再次执行模块代码,而是只返回缓存的对象,这样您就可以在第二次请求它之前看到所做的修改。

有了这些信息,在您的第一个示例中,您将在导出中返回函数,这些函数将在您每次调用它们时执行它们的代码(显然),所以如果您在函数中有一个readfile方法,那么每次都会运行它。

第二种方法通常是为了提高性能,因为代码只运行一次(在第一次需要时),每次执行导出的函数时,它们都将访问已经缓存了文件内容的变量content。你得出这个结论值得称赞:-)坚持下去。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/51655664

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档