1

我尝试使用 xlsx-populate 获取 excel 表,所以我试图从 mailEvents 集合中获取值,以便我可以在 init Collection 函数中添加或传递值。如果我已经解决了这个问题,那么我关于 excel 的问题也是解决了。在这个问题中,我想访问 init 集合中的 topHeader var。在 arr2 中有这种形式的值['open','processed']

const completeReport = (startTime,endTime) => {
    serverRef = db.collection("MailEvents");
    let getDocs = serverRef
      .where("timestamp", ">=", startTime)
      .where("timestamp", "<=", endTime)
      .get()
      .then(querySnapshot => {
        if (querySnapshot) {
          let docs = querySnapshot.docs.map(doc => doc.data());
          let arr1 = ["email", "reportName", "office"];
          let arr2 = docs.map(a => a.event);

          let topHeader = [...new Set(arr1.concat(arr2))];
        }
      });
      let query = db.collection("inits");
      let queryData = query
        .where("report", "in", ["payroll", "footprints"])

        .get()
        .then(querySnapshot => {
          if (querySnapshot) {
            let docs = querySnapshot.docs.map(doc => doc.data());
            console.log(topHeader)
          }
        });
}

所以我想要这种形式的输出

["email", "reportName", "office",'open','processed']
4

2 回答 2

2

数据是从 Firebase 异步加载的,因为它可能需要一些时间才能返回。您的主代码无需等待数据返回,而是继续立即执行。然后,当数据可用时,将then()使用该数据调用您的回调。

这意味着任何需要从 Firestore 访问数据的代码都必须在then()回调中,或者从那里调用。

例如:

const completeReport = (startTime,endTime) => {
    serverRef = db.collection("MailEvents");
    let getDocs = serverRef
      .where("timestamp", ">=", startTime)
      .where("timestamp", "<=", endTime)
      .get()
      .then(querySnapshot => {
        if (querySnapshot) {
          let docs = querySnapshot.docs.map(doc => doc.data());
          let arr1 = ["email", "reportName", "office"];
          let arr2 = docs.map(a => a.event);

          let topHeader = [...new Set(arr1.concat(arr2))];

          let query = db.collection("inits");
          let queryData = serverRef
            .where("report", "in", ["payroll", "footprints"])
            .get()
            .then(querySnapshot => {
              if (querySnapshot) {
                let docs = querySnapshot.docs.map(doc => doc.data());
                console.log(topHeader)
              }
            });
        }
      });
}

有一些替代方案,尤其是当您愿意使用更现代的 JavaScript 结构时。最简单的方法是使用async/ await关键字,它将上面的内容包装在一些对大多数开发人员来说更正常的语法糖中。

当您应用async/时,上面的代码将变成这样await

const completeReport = async (startTime,endTime) => {
    serverRef = db.collection("MailEvents");
    let querySnapshot = await serverRef
      .where("timestamp", ">=", startTime)
      .where("timestamp", "<=", endTime)
      .get();
    if (querySnapshot) {
      let docs = querySnapshot.docs.map(doc => doc.data());
      let arr1 = ["email", "reportName", "office"];
      let arr2 = docs.map(a => a.event);

      let topHeader = [...new Set(arr1.concat(arr2))];

      let query = db.collection("inits");
      querySnapshot = await serverRef
        .where("report", "in", ["payroll", "footprints"])
        .get();
      if (querySnapshot) {
        let docs = querySnapshot.docs.map(doc => doc.data());
        console.log(topHeader)
      }
    }
}

这里最大的变化是:

  • 函数上的async标记completeReport,因为调用者需要知道这个函数现在可能返回一个 promise/exhibit 异步行为。
  • await两次调用中的关键字get(),这意味着您不再需要then阻止。
  • 缩进减少,尤其是与我回答中的第一个片段相比时。

使用这种方法时要始终意识到的一件事是调用仍然是异步的。虽然使用async/await使代码更易于阅读,但它不会改变 API 的实际行为。

于 2019-12-28T16:11:50.183 回答
0

由于 let 具有块作用域,因此无法在外部访问。

建议您声明一个变量,var然后再分配它。

希望这可以帮助!

const completeReport = () => {
    let serverRef = db.collection("MailEvents");
    var topHeader; 
    let getDocs = serverRef
      .where("timestamp", ">=", startTime)
      .where("timestamp", "<=", endTime)
      .get()
      .then(querySnapshot => {
        if (querySnapshot) {
          let docs = querySnapshot.docs.map(doc => doc.data());
          let arr1 = ["email", "reportName", "office"];
          let arr2 = docs.map(a => a.event);

          topHeader = [...new Set(arr1.concat(arr2))]; //assign value
        }
      });
      serverRef = db.collection("inits");
      let queryData = serverRef
        .where("report", "in", ["payroll", "footprints"])

        .get()
        .then(querySnapshot => {
          if (querySnapshot) {
            let docs = querySnapshot.docs.map(doc => doc.data());
            console.log(topHeader)
          }
        });
}
于 2019-12-28T15:05:01.873 回答