消防功能在其他人完成javascript之后

时间:2012-04-30 03:48:35

标签: javascript jquery function

所以我试图在JavaScript中运行一个函数,但我需要所有其他函数在启动之前运行并完成。这是代码(对不起,我知道它很长,但它是我能说明发生了什么的唯一方法):

        getWeather();
        getAverage();


function getWeather() {
    $.getJSON("http://where.yahooapis.com/geocode?q=" + lat + ",+" + lon + "&gflags=R&flags=J", function(data){
        zipCode = data.ResultSet.Results[0].postal;
        zipCode = zipCode.substring(0,5);
        WOEID = data.ResultSet.Results[0].woeid;
        getYahooWeather(WOEID);
        getWeatherbug(zipCode);
        getWeatherUnderground(zipCode);
        getWorldWeather(zipCode);
    });
}

function getYahooWeather(x) {
    var query = escape('select item from weather.forecast where woeid="'+x+'"');
    var url = "http://query.yahooapis.com/v1/public/yql?q=" + query + "&format=json";


    $.getJSON(url, function(data2){
        yahooTemp = data2.query.results.channel.item.condition.temp;
        $("#yahoo-weather p").replaceWith("<p>Weather from Yahoo! powered by The Weather Channel = "+yahooTemp+"&deg;F</p>");
    });
}

function getWeatherbug(x) {
    var url = "http://i.wxbug.net/REST/Direct/GetObs.ashx?api_key="+ weatherbugAPI + "&zip="+x+"&ht=t&ic=1&f=?";
    console.log(url);

    $.ajax({
        url: url,
        dataType: "jsonp",
        success: function(data3) {
            //console.log(data3.temperature);
            wbTemp = data3.temperature;
            $("#wb-weather p").replaceWith("<p>Weather from WeatherBug = "+wbTemp+"&deg;F</p>");
        }
    });
}

function getWeatherUnderground(x) {
    $.ajax({
    url: "http://api.wunderground.com/api/b87325296cd69fa8/geolookup/conditions/q/IA/"+x+".json",
    dataType: "jsonp",
    success: function(parsed_json) {
        var location = parsed_json['location']['city'];
        wuTemp = parsed_json['current_observation']['temp_f'];
        $("#wu-weather p").replaceWith("<p>Weather from Weather Underground = "+wuTemp+"&deg;F</p>");
        }
    });
}

function getWorldWeather(x) {
    var url = "http://free.worldweatheronline.com/feed/weather.ashx?key="+wwKey+"&q="+x+"&fx=no&format=json";

    $.ajax({
        url: url,
        dataType: "jsonp",
        success: function(data6) {
            wwTemp = data6.data.current_condition[0].temp_F;
            $("#ww-weather p").replaceWith("<p>Weather from World Weather Online = "+wwTemp+"&deg;F</p>");
        }
    });
}

function getAverage() {
    avTemp = wbTemp + wwTemp + yahooTemp + wuTemp;
    console.log(avTemp);
}

我遇到的问题是每当我运行getAverage函数时它会返回NaN,因为其他函数还没有返回它们的数据。

在所有以前的函数返回数据后,有没有办法运行函数?

由于

3 个答案:

答案 0 :(得分:3)

其中一种更简单的方法是使用async之类的异步库,可以在https://github.com/caolan/async找到。您可以使用parallel函数并行运行getter,然后在所有getter完成后返回。代码最终看起来像:

async.parallel([
    function(callback){
        setTimeout(function(){
            callback(null, 'one');
        }, 200);
    },
    function(callback){
        setTimeout(function(){
            callback(null, 'two');
        }, 100);
    },
],
// optional callback
function(err, results){
    // the results array will equal ['one','two'] even though
    // the second function had a shorter timeout.
});

另一种方法是使用一个包含您正在等待的回调数的全局变量。然后在每个天气吸气剂的回调中,递减计数器。在取平均值之前,只需等到计数器为0(你可以使用setTimeout等待一段时间再重新检查)。

答案 1 :(得分:1)

要缩短内容:请在getAverage回调中拨打getJSON,即在您致电getWorldWeather(zipCode);后立即。

答案 2 :(得分:0)

确实有。您将需要执行所谓的方法传递或进行回调。

以下是未经测试的示例,说明如何进行此操作:http://jsfiddle.net/UDbeV/

让我更详细地解释一下。

var i = 0;
loopCount = function() {
    i++;
    if (i > 5) {
        getAverage();
    }
});

所以我们正在创建一个名为loopCount的函数来存储计数器。每当它被调用时,我们将其计数器增加1.

让我们将回调传递给您的方法:

    getYahooWeather(WOEID, callback);
    getWeatherbug(zipCode, callback);
    getWeatherUnderground(zipCode, callback);
    getWorldWeather(zipCode, callback);
    callback();

现在我们必须更改函数以允许第二个参数:

function getYahooWeather(x, callback) { ...

最后,在每个块的末尾,我们想要调用它:

$.getJSON(url, function(data2) {
    yahooTemp = data2.query.results.channel.item.condition.temp;
    $("#yahoo-weather p").replaceWith("<p>Weather from Yahoo! powered by The Weather Channel = " + yahooTemp + "&deg;F</p>");
    callback();
});

回调检查“它是否被调用了足够的次数(即:你在平均之前做了多少次调用),然后调用getAverage();

可能有更有效的方法来做到这一点,但它是星期一:P另外,它是未经测试的,所以如果你继续使用它,你可能需要稍微破解它。 :O)