如何处理 mongoose 模型中 find 方法的回调结果

How to work with result of the callback of the find method in a mongoose model

我正在使用猫鼬作为 nodejs-mongodb 应用程序的 ODM。

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

查看mongoose的docs可以发现:

Kitten.find(function (err, kittens) {
    if (err) return console.error(err);
    console.log(kittens);
});

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

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

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

由于我来自非函数式编程背景,所以我期待的是:

var kitten = Kitten.find({name:'Silence'}); 

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

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

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

谢谢!

编辑:我不知道为什么社区说我的问题可能与 How to return the response from an asynchronous call? 重复ODM 框架以及如何处理它也可以通过承诺完成的结果是另一回事

我想通知您,像访问数据库和检索数据这样的操作是异步操作,因此它们的工作方式不同,您不能像基本同步操作那样只分配值。事实上,这就是 Node.js 的要点。如果您关注 link,您可能会阅读更多内容: https://nodejs.org/about/

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

  • 使用回调
  • 使用承诺

回调

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

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
    }
});

请注意,由于这些异步操作的特定性,您必须在现有回调中执行所有其他操作。很快您将面临名为 'Callback Hell'.

的问题

关于回调的更多信息和 'Callback Hell' 您可以在那里阅读:

http://callbackhell.com/

承诺

这是一种更易于理解且更专业的异步函数处理方式。 Mongoose ODM 支持 promises (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
  });

那么你只需在现有的 promise 中工作,你可能会有越来越多的 '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
  });

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

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