6

我们在 nodejs 中编写服务器代码并使用 Parse javascript sdk。通常我们需要获取或更新各种 Parse 对象。例如,获取用户名为“Example”的用户。

  function fetchUser(username)
  {
      var User = Parse.Object.extend("User");
      var query = new Parse.Query("User");
      query.equalTo("username",username);

      query.first({
        success: function(results) {
        console.log("Successfully retrieved " + results.length + " scores.");
        return results;
      },
      error: function(error) {
        console.log("Error: " + error.code + " " + error.message);
      }
     });
  }

该函数可能被其他函数调用:

function test()
{
    var user = fetchUser("example");
    console.log(user); // would often print undefined
    return user;
}

function top()
{
    // some code
    var user = test();
    //some code
}

// and the function top() mmight be called by some other functions

问题是我会得到未定义的结果,因为 find/first 操作是异步运行的。有没有办法确保 fetchUser() 函数仅在查找/第一次操作成功/完成时返回?

4

2 回答 2

5

JavaScript 本质上是异步的。您必须将回调传递给fetchUser()函数并将客户端代码重写为如下所示:

function fetchUser(username, callback) {
    // do request and call callback(null, user) on success
    // or callback(some_error_object) on failure.
}

function test(callback) {
    var onFetched = function(err, user) {
        console.log(user);
        callback(err, user);
    }
    var user = fetchUser("example", onFetched);
}

您还可以使用Promise或生成器(自 EcmaScript 6 起)方法

UPD:虽然某些 API(DB 驱动程序、fs 等)允许以同步方式使用它们,但在 JavaScript 中编写代码被认为是一种糟糕的方式。由于 JS 在单个线程中运行您的代码,因此使用同步调用阻塞该线程将阻塞所有其他任务。因此,如果您开发一个同时处理几个客户端的 Web 服务器,并决定对某些 API 使用同步调用,那么一次只会为一个客户端提供服务,而其他客户端将处于挂起状态。

于 2016-11-02T13:40:19.927 回答
3

NodeJs 是单线程的,所以我们不应该阻塞进程。

在您的情况下,您有两个选择 1. 传递回调 2. 使用 Promise 库(https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise

例 1:传递回调

    function fetchUser(username, callback)
  {
      var User = Parse.Object.extend("User");
      var query = new Parse.Query("User");
      query.equalTo("username",username);

      query.first({
        success: function(results) {
        console.log("Successfully retrieved " + results.length + " scores.");
        callback(null,results)
      },
      error: function(error) {
        console.log("Error: " + error.code + " " + error.message);
        callback(error)
      }
     });
  }

客户端代码:

  function test()
    {
        var user = fetchUser("example", function(error, user){
         if(error){
          console.error(error)
         }else {
          console.log(user)
         }
      });
 }

例 2:使用 Promising 库

  function fetchUser(username){
    return new Promise(function(resolve, reject){
        var User = Parse.Object.extend("User");
        var query = new Parse.Query("User");
        query.equalTo("username",username);

        query.first({
            success: function(results) {
                console.log("Successfully retrieved " + results.length + " scores.");
                resolve(results)
            },
            error: function(error) {
                console.log("Error: " + error.code + " " + error.message);
                reject(error)
            }
        });
    })
}

客户端代码:

function test()
    {
      fetchUser("example")
        .then(function(user){
          console.log(user)  
        })
        .catch(function(error){
         console.error(error)
        })  
 }
于 2016-11-02T13:56:48.397 回答