I want to do a for each loop but have it run synchronously. Each iteration of the loop will do an http.get call and that will return json for it to insert the values into a database. The problem is that the for loop runs asynchronously and that causes all of the http.gets to all run at once and my database doesn't end up inserting all of the data.I am using async-foreach to try to do what I want it to do, but I don't have to use it if I can do it the right way.
mCardImport = require('m_cardImport.js'); var http = require('http'); app.get('/path/hi', function(req, res) { mCardImport.getList(function(sets) { forEach(sets, function(item, index, arr) { theUrl = 'http://' + sets.set_code + '.json'; http.get(theUrl, function(res) { var jsonData = ''; res.on('data', function(chunk) { jsonData += chunk; }); res.on('end', function() { var theResponse = JSON.parse(jsonData); mCardImport.importResponse(theResponse.list, theResponse.code, function(theSet) { console.log("SET: " + theSet); }); }); }); }); }); });
and my model
exports.importResponse = function(cardList, setCode, callback) { mysqlLib.getConnection(function(err, connection) { forEach(cardList, function(item, index, arr) { var theSql = "INSERT INTO table (name, code, multid, collector_set_num) VALUES " + "(?, ?, ?, ?) ON DUPLICATE KEY UPDATE id=id"; connection.query(theSql, [item.name, setCode, item.multid, item.number], function(err, results) { if (err) { console.log(err); }; }); }); }); callback(setCode); };
NodeJS is an asynchronous event-driven JavaScript runtime environment designed to build scalable network applications. Asynchronous here refers to all those functions in JavaScript that are processed in the background without blocking any other request.
Asynchronous functions are generally preferred over synchronous functions as they do not block the execution of the program whereas synchronous functions block the execution of the program until it has finished processing. Some of the asynchronous methods of fs module in NodeJS are: fs.
for loop is synchronous. B should not be executed before for loop completes.
js. JavaScript is asynchronous in nature and so is Node. Asynchronous programming is a design pattern which ensures the non-blocking code execution.
With recursion the code is pretty clean. Wait for the http response to come back then fire off next attempt. This will work in all versions of node.
var urls = ['http://stackoverflow.com/', 'http://security.stackexchange.com/', 'http://unix.stackexchange.com/']; var processItems = function(x){ if( x < urls.length ) { http.get(urls[x], function(res) { // add some code here to process the response processItems(x+1); }); } }; processItems(0);
A solution using promises would also work well, and is more terse. For example, if you have a version of get that returns a promise and Node v7.6+, you could write an async/await function like this example, which uses some new JS features.
const urls = ['http://stackoverflow.com/', 'http://security.stackexchange.com/', 'http://unix.stackexchange.com/']; async function processItems(urls){ for(const url of urls) { const response = await promisifiedHttpGet(url); // add some code here to process the response. } }; processItems(urls);
Note: both of these examples skip over error handling, but you should probably have that in a production app.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With