如何在猫鼬模型中处理 find 方法的回调结果

Jor*_*rza 2 callback mongodb node.js

我使用 mongoose 作为 nodejs-mongodb 应用程序的 ODM。

是我的第一个 nodejs 应用程序,我来自非函数式编程背景。

查看猫鼬的文档,您可以找到:

Kitten.find(function (err, kittens) {
    if (err) return console.error(err);
    console.log(kittens);
});
Run Code Online (Sandbox Code Playgroud)

太好了,在这里我们将 find 函数作为我们模型(Kitten)的一部分,它实际上可以找到文档并在回调函数中将其检索为“kittens”,在这种情况下它使用 console.log() 。

但是我想知道这个函数式编程是如何用来将这个值分配给一个变量的(因为我在models文件夹的另一个文件中有这个,并且我用require导入了模块)

我发现了另一个关于类似要求 ObjectId 的问题,但是当您使用回调仅使用 console.log 打印答案时,它们提供了相同类型的答案,如果我们说实话,这没有用。

由于我来自非函数式编程背景,因此我期待以下内容:

var kitten = Kitten.find({name:'Silence'}); 
Run Code Online (Sandbox Code Playgroud)

据我所知,如果你在回调中分配一个新变量,变量的范围就在回调函数内,与返回变量相同,甚至在方法不起作用之前声明一个变量。

我确定我缺少某些东西。这个项目太大了,我认为他们不会忘记提供一种方法来做到这一点。我认为函数式编程中有一些我遗漏或我不知道的东西。

那么,如何才能做到这一点?

谢谢 !

编辑:我不知道为什么社区会说我的问题可能是How to return the response from an asynchronous call? 这个问题一般更面向 js ajax async 并且这个问题面向 ODM 框架以及如何处理它也可以通过承诺完成的结果,这是另一回事

Rom*_*nko 6

我想通知您,诸如访问数据库和检索数据之类的操作是异步操作,因此它们的工作方式不同,您不能像在基本同步操作中那样只分配值。事实上,这是 Node.js 的重点。如果您点击链接,您可以阅读更多内容:https : //nodejs.org/about/

那么如何解决您遇到的问题。我可以为您提供2种方式:

  • 使用回调
  • 使用承诺

回调

这实际上就是你目前正在使用的。您需要运行基本函数并将另一个(回调)函数作为附加参数放在那里,因此一旦一切准备就绪,回调函数将被触发,您将能够获得结果。如何将结果接收到您的变量中:

var searchQuery = {}; //or something specific like {name: 'Kitten Name'};
var foundKittens = [];
Kitten.find(searchQuery , function (err, kittens) {
    if (err) {
            //do something with error
    } else {
        foundKittens = kittens
    }
});
Run Code Online (Sandbox Code Playgroud)

请注意,由于这些异步操作的特殊性,您必须在现有回调内进行所有其余操作。很快,您将面临称为“回调地狱”的问题。

有关回调和“回调地狱”的更多信息,您可以在此处阅读:

http://callbackhell.com/

承诺

这是使用异步函数更容易理解和专业的方式。Mongoose ODM 支持承诺 ( http://mongoosejs.com/docs/api.html#promise_Promise ),因此您可以执行以下操作。

var searchQuery = {}; //or something specific like {name: 'Kitten Name'};
var foundKittens = [];

Kitten
  .find(searchQuery)
  .exec()
  .then(function(kittens){
      //here you can assign result value to your variable
      //but this action is useless as you are working with results directly
      foundKittens = kittens;
      return kittens;
  })
  .onReject(function(err){
      throw err; //or something else
  });
Run Code Online (Sandbox Code Playgroud)

因此,您只需在现有承诺内工作,您可能会有越来越多的“then”语句,例如:

Kitten
  .find(searchQuery)
  .exec()
  .then(function(kittens){
      foundKittens = kittens;
      return kittens;
  })
  .then(countKittents)
  .then(changeSomethingInKittents)
  .then(saveKittentsToDb)
  .onReject(function(err){
      throw err; //or something else
  });
Run Code Online (Sandbox Code Playgroud)

有关承诺的更多信息,请阅读:

http://pouchdb.com/2015/05/18/we-have-a-problem-with-promises.html