0

因此,在我的应用程序中,我有一个按钮可以与从 Internet 下载一些数据并对其进行过滤的库进行对话。当应用程序执行此操作时,屏幕会冻结,并且在用户看来就像应用程序崩溃了一样。但事实并非如此,因为它正在下载数据。

这是我的代码:

GetDetailsBtn.TouchUpInside += (sender, e) => {

    var defaults = NSUserDefaults.StandardUserDefaults;

    if (RefNr.Text != string.Empty && RefNr.Text != null) {

        FilteredDataRef = _FetchingData.getTrackTraceData (defaults.StringForKey ("SecurityToken"), RefNr.Text);

        if (FilteredDataRef == null) {
            UIAlertView InvalidAlert = new UIAlertView ("Reference number invalid", "The reference number that you have entered is not linked to the current security code. You can change your security code in the settings.", null, "OK", null);
            InvalidAlert.Show ();

        } else {
            FilteredDataReceived = _FetchingData.FilteringOnReceived (FilteredDataRef);

            FilteredDataPlanned = _FetchingData.FilteringOnPlanned (FilteredDataRef);

            FilteredDataLoadingETA = _FetchingData.FilteringOnLoadingETA (FilteredDataRef);

            FilteredDataLoadingFinal = _FetchingData.FilteringOnLoadingFinal (FilteredDataRef);

            FilteredDataUnloadingETA = _FetchingData.FilteringOnUnloadingETA (FilteredDataRef);

            FilteredDataUnloadingFinal = _FetchingData.FilteringOnUnloadingFinal (FilteredDataRef);   

            this.PerformSegue (MoveToTrackTraceDetailsSegue, this);

            //foreach (string s in FilteredDataPlanned)
            //    Console.WriteLine (s);

        }

    } else {
        UIAlertView InvalidAlert = new UIAlertView ("Reference number cannot be empty", "You did not provide a reference number. We need your reference number to trace identify the shipment you would like to trace.", null, "OK", null);
        InvalidAlert.Show ();
    }
};

数据下载:

public IEnumerable<string> getTrackTraceData (string securityCode, string referenceNumber)
{
    WebRequest request = WebRequest.Create ("http://plex.janssen1877.com/app/life/" + securityCode);

    HttpWebResponse response = (HttpWebResponse)request.GetResponse ();

    Stream dataStream = response.GetResponseStream ();

    StreamReader reader = new StreamReader (dataStream);

    string FetchedData = reader.ReadToEnd ();

    reader.Close ();

    dataStream.Close ();

    response.Close ();

    var FetchingDataItems = FetchedData.Split (new char[] { '\n' });

    if (FetchingDataItems != null) {
        var filteredResult = FetchingDataItems.Where (x => x.Contains (referenceNumber));

        return filteredResult;
    } else {
        return null;
    }
}

现在我想使用一个名为BTProgressHUD. 这只是一个花哨的微调器。我认为,如果我将按钮操作放在BTProgressHUD.show();按钮操作的顶部和BTProgressHUD.Dismiss();按钮上,它将在加载开始时显示并在加载完成时关闭。

不是这种情况。它在新的视图控制器中显示得非常快,并在一秒钟内再次消失。我究竟做错了什么?

编辑例如:

public IEnumerable<string> getTrackTraceData (string securityCode, string referenceNumber)
        {


            string url = string.Format ("http://plex.janssen1877.com/app/life/" + securityCode);

            HttpWebRequest HttpRequest = (HttpWebRequest)WebRequest.Create (url);

            string FetchedData = new StreamReader (HttpRequest.GetResponse ().GetResponseStream ()).ReadToEnd ();


            var FetchingDataItems = FetchedData.Split (new char[] { '\n' });

            if (FetchingDataItems != null) {
                var filteredResult = FetchingDataItems.Where (x => x.Contains (referenceNumber));

                return filteredResult;
            } else {
                return null;
            }


        }
4

2 回答 2

3

弗洛里安,

根据 .NET 文档和HttpWebRequest GetResponse。如何等待响应?您需要以异步方式执行下载(可能还有解析)。

您的实际应用程序的行为是正确的。当您在主线程中执行同步请求时,您会冻结应用程序,因此 UI 元素不会更新。主线程以串行方式处理执行。

为避免这种情况,您有两种不同的解决方案。一方面,您需要转向异步请求。另一方面,您可以使用同步请求创建后台线程(不同的执行路径)。我更喜欢前者。因此,例如,在启动异步请求后,显示指示器。完成后(在回调中),关闭指示器并执行 segue。

例如,您可以按照以下关于如何实现此目的的讨论:如何异步使用 HttpWebRequest (.NET)?.

要了解主线程(和运行循环)是如何工作的,我建议阅读有关NSRunLoop 的弹簧棒

希望有帮助。

编辑

您应该使用如下模式(来源How to use HttpWebRequest (.NET) asynchronously?):

HttpWebRequest webRequest;

void StartWebRequest()
{
    webRequest.BeginGetResponse(new AsyncCallback(FinishWebRequest), null);
}

void FinishWebRequest(IAsyncResult result)
{
    webRequest.EndGetResponse(result);
}
于 2013-04-16T20:59:51.540 回答
1

听起来您正试图在与 UI 相同的线程中执行所有这些操作,这就是为什么您的应用程序冻结并等待处理完成而不显示任何内容的原因。我会在某种后台工作人员中执行此下载操作。然后我不确定在 Mono 中,但在前后访问主线程并显示和关闭您的加载组件。

于 2013-04-16T21:06:55.797 回答