2016-04-07 3 views
1

У меня проблема с синхронизацией с использованием fs.readfile, они не вводятся в заблуждение в первый раз, если я снова попробую запрос, тогда у меня есть результаты. Я знаю, что могу использовать обещание, и я нахожу что-то полезное с Q from Kriskowal. Я пробовал, но безуспешно. Я не очень хорошо понял, как его реализовать. Если кто-то может помочь в этом, я буду бесконечно благодарен.Обещают, как реализовать в этой ситуации?

код:

"use strict" 

var Q = require('q'); 
var fs = require('fs'); 
var arrayZipBand = []; 
var jsonZipCode = []; 
var arrayParsedZipcodeBr = []; 

exports.parse = function(opts) { 

if (opts.zipBand.constructor != Array) { 
opts.zipBand = [opts.zipBand]; 
} 
if (opts.location.constructor != Array) { 
opts.location = [opts.location]; 
} 

if (opts.config === 0) { 
opts.zipBand.forEach(function(file) { 
    fs.readFile(file, 'utf8', function(err, logData) { 
    if (err) throw err; 
    let text = logData.toString(); 
    decodeURIComponent(text); 

    let lines = text.split('\n'); 

    lines.forEach(function(line) { 
     let parts = line.split('@'); 

     if (parts[1] != undefined) { 
     let obJson = { 
      LOC_NU: parts[0], 
      LOC_CEP_INI: parts[1], 
      LOC_CEP_FIM: parts[2] 

     } 
     arrayZipBand.push(obJson); 
     } 

    }); 


    }); 

}); 

opts.location.forEach(function(file) { 
    fs.readFile(file, 'utf8', function(err, logData) { 


    if (err) throw err; 

    let text = logData.toString(); 
    decodeURIComponent(text); 

    let lines = text.split('\n'); 

    lines.forEach(function(line) { 
     let parts = line.split('@'); 

     if (parts[1] != undefined) { 
     for (let i = 0; i < arrayZipBand.length; i++) { 
      if (parts[0] == arrayZipBand[i].LOC_NU) { 
      jsonZipCode.push(arrayZipBand[i]); 
      } 
     } 
     if (jsonZipCode === undefined) { 
      throw "Was not possible to find Zipcode for the id " + parts[0]; 
     } 

     for (let i = 0; i < jsonZipCode.length; i++) { 
      let obJson = { 
      LOC_NU: parts[0], 
      UFE_SG: parts[1], 
      LOC_NO: parts[2], 
      MUN_NU: parts[8], 
      LOC_CEP_INI: jsonZipCode[i].LOC_CEP_INI, 
      LOC_CEP_FIM: jsonZipCode[i].LOC_CEP_FIM 
      } 

      arrayParsedZipcodeBr.push(obJson); 
     } 
     jsonZipCode = []; 

     } 


    }); 
    }); 
}); 



}; 
return arrayParsedZipcodeBr; 

} 
+1

https://www.npmjs.com/package/promise-denodeify –

+1

Ну, как вы пытаетесь использовать обещание? Пожалуйста, отредактируйте свой вопрос, чтобы показать нам вашу попытку. Вы читали Q-документы? Вы видели пример вызова методов 'fs' с помощью вспомогательных функций' Q', чтобы вы получили от них обещания? – Bergi

ответ

2

хмм, кажется, вы пытаетесь читать файлы в цикле петлевой то, но в асинхронном способом. Первый вопрос, почему асинхронное чтение этих файлов? Вы всегда можете прочитать их в синхронизации образом:

var data=fs.readFileSync(fname, encoding); 

Кстати, если вы хотите, чтобы прочитать их асинхронными и сохранить для ... петель вам нужно что-то вроде обещания, или Timed Wait или более сложного механизм синхронизации.

Вы можете держать его просто, без использования каких-либо других пакетов/модулей таким образом:

/** 
     * Promise.All 
     * @param items Array of objects 
     * @param block Function block(item,index,resolve,reject) 
     * @param done Function Success block 
     * @param fail Function Failure block 
     * @example 

      promiseAll(["a","b","c"], 
      function(item,index,resolve,reject) { 
      MyApp.call(item,function(result) { 
       resolve(result); 
      }, 
      function(error) { reject(error); }): 
      }, 
      function(result) { // aggregated results 

      },function(error) { // error 

      }) 

     * @author Loreto Parisi (loretoparisi at gmail dot com) 
     */ 
    promiseAll: function(items, block, done, fail) { 
      var self = this; 
      var promises = [], 
       index = 0; 
      items.forEach(function(item) { 
       promises.push(function(item, i) { 
        return new Promise(function(resolve, reject) { 
         if (block) { 
          block.apply(this, [item, index, resolve, reject]); 
         } 
        }); 
       }(item, ++index)) 
      }); 
      Promise.all(promises).then(function AcceptHandler(results) { 
       if (done) done(results); 
      }, function ErrorHandler(error) { 
       if (fail) fail(error); 
      }); 
     }, //promiseAll 

так что вы можете назвать его как

promiseAll(arrayOfItems, function(item, index, resolve, reject) { 
    // do something on item 
    if (someSuccessCondOnThisItem) { 
     resolve(item) 
    } else { 
     reject(new Error("operation failed")) 
    } 
}, function(results) { // aggregated results 

    console.log("All done %d", results.length); 

}, function(error) { // error 
    console.log(error.toString()); 
}); 

Имейте в виду, что это очень упрощенный подход, но в большинстве случаев он работает при циклическом перемещении по массивам.

Вот простой рабочий пример на детской площадке:

var console = { 
 
log : function(s) { document.getElementById("console").innerHTML+=s+"<br/>"} 
 
} 
 
var promiseAll= function(items, block, done, fail) { 
 
      var self = this; 
 
      var promises = [], 
 
       index = 0; 
 
      items.forEach(function(item) { 
 
       promises.push(function(item, i) { 
 
        return new Promise(function(resolve, reject) { 
 
         if (block) { 
 
          block.apply(this, [item, index, resolve, reject]); 
 
         } 
 
        }); 
 
       }(item, ++index)) 
 
      }); 
 
      Promise.all(promises).then(function AcceptHandler(results) { 
 
       if (done) done(results); 
 
      }, function ErrorHandler(error) { 
 
       if (fail) fail(error); 
 
      }); 
 
     }; //promiseAll 
 

 
arr=[1,2,3] 
 
promiseAll(arr 
 
       ,function(item,index,resolve,reject) { 
 
        console.log("Resolving item[" + index+"]") 
 
        var okCond=true 
 
        if(okCond) {resolve(item)} else { reject(new Error("item[" + index+"]")) } 
 
       } 
 
       ,function(results) { // aggregated results 
 
console.log("All done of "+results.length); 
 
       } 
 
       ,function(error) { // error 
 
       console.log(error); 
 
});
<div id="console"/>

Наконец, полный асинхронный пример, показывающий, как отложить выполнение XMLHttpRequest, при езде на велосипеде по списку. ExecutionBlock вызывает reject и resolve после ответа SimpleRequest, в результате чего Promise ожидает его выполнения до вызова then.

var console = { 
 
    log: function(s) { 
 
     document.getElementById("console").innerHTML += s + "<br/>" 
 
    } 
 
    } 
 
    // Simple XMLHttpRequest 
 
    // based on https://davidwalsh.name/xmlhttprequest 
 
SimpleRequest = { 
 
    call: function(what, response) { 
 
     var request; 
 
     if (window.XMLHttpRequest) { // Mozilla, Safari, ... 
 
     request = new XMLHttpRequest(); 
 
     } else if (window.ActiveXObject) { // IE 
 
     try { 
 
      request = new ActiveXObject('Msxml2.XMLHTTP'); 
 
     } catch (e) { 
 
      try { 
 
      request = new ActiveXObject('Microsoft.XMLHTTP'); 
 
      } catch (e) {} 
 
     } 
 
     } 
 
     // state changes 
 
     request.onreadystatechange = function() { 
 
     if (request.readyState === 4) { // done 
 
      if (request.status === 200) { // complete \t 
 
      response(request.responseText) 
 
      } else response(); 
 
     } 
 
     } 
 
     request.open('GET', what, true); 
 
     request.send(null); 
 
    } 
 
    } 
 
    //PromiseAll 
 
var promiseAll = function(items, block, done, fail) { 
 
    var self = this; 
 
    var promises = [], 
 
    index = 0; 
 
    items.forEach(function(item) { 
 
    promises.push(function(item, i) { 
 
     return new Promise(function(resolve, reject) { 
 
     if (block) { 
 
      block.apply(this, [item, index, resolve, reject]); 
 
     } 
 
     }); 
 
    }(item, ++index)) 
 
    }); 
 
    Promise.all(promises).then(function AcceptHandler(results) { 
 
    if (done) done(results); 
 
    }, function ErrorHandler(error) { 
 
    if (fail) fail(error); 
 
    }); 
 
}; //promiseAll 
 

 
// LP: deferred execution block 
 
var ExecutionBlock = function(item, index, resolve, reject) { 
 
    SimpleRequest.call('https://icanhazip.com/', function(result) { 
 
    if (result) { 
 
     console.log("Response[" + index + "] " + result); 
 
     resolve(result); 
 
    } else { 
 
     reject(new Error("call error")); 
 
    } 
 
    }) 
 
} 
 

 
arr = [1, 2, 3] 
 
promiseAll(arr, function(item, index, resolve, reject) { 
 
    console.log("Making request [" + index + "]") 
 
    ExecutionBlock(item, index, resolve, reject); 
 
}, function(results) { // aggregated results 
 
    console.log("All response received " + results.length); 
 
    console.log(JSON.stringify(results)); 
 
}, function(error) { // error 
 
    console.log(error); 
 
});
<div id="console" />

+1

спасибо за примеры @loretoparisi, я многому научился – michelpm1

+0

@ michelpm1 приветствую вас, может быть, вы можете отредактировать название в чем-то более показательном, как обещание для цикла и т. Д. Последний пример показывает, как уменьшить массив как ' XMLHttpRequest' на веб-сайты url ... – loretoparisi

 Смежные вопросы

  • Нет связанных вопросов^_^