2

我有一门课,它从气象局或气象部门收集 7 天的预报数据,并将其呈现在网页上。脚本每 30 分钟运行一次,以从局获取更新数据。

该局以制表符分隔格式提供数据,并带有标题行。提取所有字段后,我将值放入 aDictionary<string,string>中进行解析。出于显而易见的原因,组织数据的重要字段是“forecast_date”。所以在我开始解析之前,我确保我的字典确实包含这个键。

这是我正在做的一个非常简化的示例:

static object ForecastLockingObj= new object();
private void UpdateWeather()
{
    if(isTimeForUpdate())
    {
        lock(ForecastLockingObj)
        {
            if(isTimeForUpdate())
            {
                Dictionary<string, string> ForecastData = Get7DayForecast();
                int forecastDate = int.MinValue;
                if (ForecastData.ContainsKey("forecast_date") && int.TryParse(ForecastData["forecast_date"], out forecastDate))
                {
                    //Parse the data
                    SetNextUpdateTime();
                }
            }
        }
    }
}

这实际上在大多数情况下都有效。但偶尔我会遇到以下异常:

[KeyNotFoundException: The given key was not present in the dictionary.]
   System.ThrowHelper.ThrowKeyNotFoundException() +28
   System.Collections.Generic.Dictionary`2.get_Item(TKey key) +7457036
   CoA.WebUI.Controls.WeatherWidget.UpdateWeather() in C:\dev\WeatherWidget.cs:231

其中第 231 行是检查“forecast_date”是否存在的 if 语句,然后尝试将其解析为整数。笔记; 该服务可靠地将日期呈现为整数(例如 20130515),因此这更像是一种健全性检查。

ContainsKey不应该抛出这个异常,所以我觉得它一定是我ForecastData["forecast_date"]在 TryParse 中引用的地方。

我的问题是这个;当然,如果 ContainsKey 返回 false,则 TryParse 不应该运行。那么为什么它会在一个语句中报告一个键的存在,然后在下一个语句中否认它的存在......而我们在一个锁中并且我们正在处理的字典是非静态的和本地的?

作为旁白; 这通常发生在下午,当无线电通信局发布下一个长期预测时。例外情况发生在一些页面加载,然后是权利本身。


这是完整的 Get7DayForecast 方法

private Dictionary<string, string> Get7DayForecast()
{
    int linenumber = 0;
    int locationNameKey = 0;
    List<string> keys = new List<string>();
    Dictionary<string, string> myLocationData = new Dictionary<string, string>();

    FtpWebRequest ftp = (FtpWebRequest)WebRequest.Create(ForecastURL);
    ftp.Method = WebRequestMethods.Ftp.DownloadFile;
    ftp.Credentials = new NetworkCredential();

    FtpWebResponse ftp_response = (FtpWebResponse)ftp.GetResponse();

    if (ftp_response.WelcomeMessage.StartsWith("230") && ftp_response.StatusDescription.StartsWith("150"))
    {
        Stream ftp_responseStream = ftp_response.GetResponseStream();
        StreamReader ftp_reader = new StreamReader(ftp_responseStream);

        while (ftp_reader.Peek() >= 0)
        {
            linenumber++;
            string line = ftp_reader.ReadLine();
            List<string> temp = (List<string>)line.Split(ForecastDelimiter).ToList<string>();
            if (linenumber == 1)
            {
                //Break if the header line does not contain the fields we require
                if (!ForecastRequiredFields.All(line.Contains)) { break; }
                keys = temp;
                locationNameKey = keys.IndexOf(ForecastLocationFieldName);
            }
            else if (temp.Count == keys.Count && temp[locationNameKey] == ForecastLocationName)
            {
                for (int i = 0; i < keys.Count; i++)
                {
                    myLocationData.Add(keys[i], temp[i]);
                }
                //Break if we've just parsed the data we were looking for
                break;
            }
        }

        ftp_reader.Close();
    }
    ftp_response.Close();
    return myLocationData;
}
4

1 回答 1

2

老实说,我不明白为什么你的代码会失败,但你应该考虑利用Trace看看发生了什么。使用它也不会受到伤害TryGetValue

var map = Get7DayForecast();

string forecastDateString;
if (!map.TryGetValue("forecast_date", out forecastDateString))
{
    Trace.WriteLine("forecast_date entry was not found.");
    return;
}

int foreCastDate;
if (!int.TryParse(forecastDateString, out foreCastDate)
{
    Trace.WriteLine("Value was not a valid integer: " + forecastDateString);
    return;
}

SetNextUpdateTime();
于 2013-05-15T01:24:40.480 回答