我有一个网站的以下简单且简短的javascript代码:
db.collection("classes").doc(data.readGrades).get().then(function(doc) {
if (doc.exists) {
const data = doc.data();
const members = data.members;
members.forEach(el => {
db.collection("users").doc(el).collection("grades").get().then(function(querySnapshot) {
querySnapshot.forEach(function(doc) {
const data = doc.data();
//some stuff
});
});
})
} else {
console.log("No such document!");
}
}).catch(function(error) {
console.log("Error getting document:",error);
});
如您所见,我需要运行两个Firebase Firestore任务.重要的是首先要运行db.collection(“ classes”).doc(data.readGrades).get().then(function(doc){,并且对于每个成员,此任务都应遵循db.collection(“ users” ).doc(el).collection(“ grades”).get().then(function(querySnapshot){(注意:第二个任务由于变量而需要第一个).所以我的问题是我必须等待在下一个成员的循环可以继续之前完成第二项任务.
而这恰恰是我的问题:在为下一个成员运行循环之前如何完成第二个任务?
db.collection("classes").doc(data.readGrades).get().then(function(doc) {
if (doc.exists) {
const data = doc.data();
const members = data.members;
members.reduce((chain,el) => {
table_number++;
const html = fillTemplate("grade_table" + table_number,member_name);
document.getElementById("main_padding").insertAdjacentHTML('beforeend',html);
return chain.then(() =>
db.collection("users").doc(el).collection("grades").get().then(function(querySnapshot) {
querySnapshot.forEach(function(doc) {
const data = doc.data();
addToTable("grade_table" + table_number,doc.id,data.mdl,data.klu);
});
})
)
},Promise.resolve());
} else {
console.log("No such document!");
}
}).catch(function(error) {
console.log("Error getting document:",error);
});
function fillTemplate(table_name,id) {
console.log("ID OF TABLES " + table_name);
return `
<div class="noten_tabelle_permission" id="noten_tabelle_permission">
<h1 id="member_name">${id}</h1>
<table id="${table_name}" style="width:100%">
<tr>
<th>Fach</th>
<th>mündlich</th>
<th>Klausur</th>
</tr>
<!-- Make content with js code -->
</table>
</div>
`;
}
function addToTable(table_name,subject,mdl,klu) {
var subject_name = getSubjectByNumber(subject);
var short_subject = getSubjectShortByNumber(subject);
//Zeile erstellen
console.log("addToTable " + table_name);
var y = document.createElement([short_subject]);
y.setAttribute("id",[short_subject]);
document.getElementById([table_name]).appendChild(y);
//Spalten in einer Zeile
var y = document.createElement("TR");
y.setAttribute("id",[short_subject]);
//Spalten in einer Zeile
var cE = document.createElement("TD");
var tE = document.createTextNode([subject_name]);
cE.appendChild(tE);
y.appendChild(cE);
var a = document.createElement("TD");
var b = document.createTextNode([mdl]);
a.appendChild(b);
y.appendChild(a);
var c = document.createElement("TD");
var d = document.createTextNode([klu]);
c.appendChild(d);
y.appendChild(c);
document.getElementById(table_name).appendChild(y);
}
因此,诺言行得通.但是现在我有一个问题,就是诺言行不通.如您所见,我想为每个成员运行一个Firestore(Database)任务.因此,诺言应解雇每个新成员.但是,诺言最终会首先兑现.
背景知识:我从数据库的ArrayList中获取成员,对于每个单个成员,我都需要基于该成员获取特定的值.加载成员的值后,我将它们添加到表中,该表将填充上一个任务中的每个成员(如您所见).正如我所说的,诺言最终将兑现.因此,这些值将不会添加到成员的特定表中.它们将被添加到最后一个表中.
编辑2:
这是编辑后的代码:
members.reduce((chain,el) => {
db.collection("users").doc(el).collection("user_data").doc("u").get().then(function (doc) {
const data = doc.data();
var member_name = data.name;
table_number++;
const html = fillTemplate("grade_table" + table_number,member_name);
document.getElementById("main_padding").insertAdjacentHTML('beforeend',html);
})
return chain.then(((table_num) => {
return db.collection("users").doc(el).collection("grades").get().then(function (querySnapshot) {
querySnapshot.forEach(function (doc) {
const data = doc.data();
addToTable("grade_table" + table_num,data.klu);
});
})
}).bind(null,table_number))
},Promise.resolve());
所以这里的问题是,我获得member_name的任务异步运行,而另一部分则不然.因此,只有在加载了所有内容后,我才能运行整个任务.因为如您所见,我还需要member_name.因此,如何重新编码,以便仅加载所有内容(member_name / grades)时任务运行.
db.collection("classes").doc(data.readGrades).get().then(function(doc) {
if (doc.exists) {
const data = doc.data();
const members = data.members;
members.reduce((chain,el) => {
return chain.then(() =>
db.collection("users").doc(el).collection("grades").get().then(function(querySnapshot) {
querySnapshot.forEach(function(doc) {
const data = doc.data();
//some stuff
});
})
)
},error);
});
更新#1:
考虑到更新的问题,数据可能会添加到最后一个表中,因为在答应解决期间,所有reduce迭代均已完成(因此table_name变量保留了最后一次迭代的值).您可以通过将table_name值绑定到在promise决议上调用的回调来解决它:
members.reduce((chain,el) => {
table_number++;
const html = fillTemplate("grade_table" + table_number,doc.id);
document.getElementById("main_padding").insertAdjacentHTML('beforeend',html);
return chain.then(((table_num) => {
return db.collection("users").doc(el).collection("grades").get().then(function (querySnapshot) {
querySnapshot.forEach(function (doc) {
const data = doc.data();
addToTable("grade_table" + table_num,data.klu);
});
})
}).bind(null,table_number))
},Promise.resolve());
更新#2:
为了随后在单个reduce迭代中运行多个异步操作,请尝试使用Promise的then方法将它们链接起来:
members.reduce((chain,el) => {
return chain
.then(() => db.collection("users").doc(el).collection("user_data").doc("u").get())
.then(doc => {
const data = doc.data();
var member_name = data.name;
table_number++;
const html = fillTemplate("grade_table" + table_number,member_name);
document.getElementById("main_padding").insertAdjacentHTML('beforeend',html);
return table_number;
})
.then(table_num => {
return db.collection("users").doc(el).collection("grades").get().then(function (querySnapshot) {
querySnapshot.forEach(function (doc) {
const data = doc.data();
addToTable("grade_table" + table_num,data.klu);
});
})
})
},Promise.resolve());