MongoDB
 sql >> Datenbank >  >> NoSQL >> MongoDB

Umgang mit Rückrufen in einer for-Schleife (Node.JS)

Dieses Problem wird als "callback hell" bezeichnet . Es gibt viele andere Ansätze wie die Verwendung von Promise und Async Bibliotheken, die Sie finden werden.

Ich freue mich mehr auf den nativen async ES7 bringen wird, die Sie tatsächlich heute mit der Transpiler-Bibliothek Babel verwenden können .

Aber der bei weitem einfachste Ansatz, den ich gefunden habe, ist der folgende:Sie nehmen die langen Callback-Funktionen heraus und definieren sie außerhalb.

router.route('/report') // the REST api address
    .post(calling_a_POST)

function calling_a_POST(req, res) {
    ...
    var data = "";
    https.get(url, function callback(response) {
        ...
        response.on("end", response_on_end_callback); // --> take out
        response.on("error", console.error);
    });
}

function response_on_end_callback() {                 // <-- define here
    ...
    for (var i = 0; i < length; i++) {
        var report = new Report(array.pop());
        ...
        Report.find({ id: report['id'] })
              .count(Report_find_count_callback);     // --> take out
    };
    res.json({
        message: 'Grabbed Report'
    });
}

function Report_find_count_callback(err, count) {     // <-- define here
    ...
    if (count == 0) {
        report.save(function(err) {                   // !! report is undefined here
            console.log('saved');
            if (err)
                res.send(err);                        // !! res is undefined here
        });
    }
}

Ein Vorbehalt ist, dass Sie nicht auf alle Variablen innerhalb des ehemaligen Callbacks zugreifen können, da Sie sie aus dem Gültigkeitsbereich genommen haben.

Dies könnte mit einer Art „Dependency Injection“-Wrapper gelöst werden, um die erforderlichen Variablen zu übergeben.

router.route('/report') // the REST api address
    .post(calling_a_POST)

function calling_a_POST(req, res) {
    ...
    var data = "";
    https.get(url, function callback(response) {
        ...
        response.on("end", function(err, data){       // take these arguments
            response_on_end(err, data, res);          // plus the needed variables
        });
        response.on("error", console.error);
    });
}

function response_on_end(err, data, res) {  // and pass them to function defined outside
    ...
    for (var i = 0; i < length; i++) {
        var report = new Report(array.pop());
        ...
        Report.find({ id: report['id'] })
            .count(function(err, count){
                Report_find_count(err, count, report, res);  // same here
            });
    };
    res.json({                                        // res is now available
        message: 'Grabbed Report'
    });
}

function Report_find_count(err, count, report, res) {        // same here
    ...
    if (count == 0) {
        report.save(function(err) {                   // report is now available
            console.log('saved');
            if (err)
                res.send(err);                        // res is now available
        });
    }
}

Mir ist klar, dass ich hier einen Fehler gemacht habe:

function calling_a_POST(req, res) {
    ...
    var data = "";
    https.get(url, function callback(response) {
        ...
        //sponse.on("end", function(err, data){
        response.on("end", function(err){ // data shouldn't be here
            response_on_end(err, data, res);
        });
        response.on("error", console.error);
    });
}

Ein weiteres Problem, das ich vorhersehen könnte, das hier vielleicht nicht auftaucht, aber trotzdem besser zu besprechen wäre. Die data Da es sich um eine Zeichenfolge handelt, die im Gegensatz zu einem Objekt ein primitiver Typ ist, wird sie "als Wert übergeben". Weitere Informationen

Es ist besser, die Variable in ein Objekt zu packen und das Objekt zu übergeben, da Objekte in Javascript immer "per Referenz übergeben" werden.

function calling_a_POST(req, res) {
    ...
    // var data = ""; // 
    var data_wrapper = {};
    data_wrapper.data = {};                                // wrap it in an object
    https.get(url, function callback(response) {
        ...
        response.on("data", function(chunk){
            data_wrapper.data += chunk.toString() + "";   // use the dot notation to reference
        });
        response.on("end", function(err){ 
            response_on_end(err, data_wrapper, res);      // and pass that object
        });
        response.on("error", console.error);
    });
}

function response_on_end_callback(err, data_wrapper, res) {
    var data = data_wrapper.data;                         // later redefine the variable
    ...
    for (var i = 0; i < length; i++) {
        var report = new Report(array.pop());
        ...