指示仅在匿名事件处理程序完成后返回的方法

时间:2015-10-16 18:45:49

标签: c#

在静态方法中我正在使用WebBrowser。我正在订阅它的Navigated事件匿名方法。我可以指示我的方法仅在WebBroswer触发Navigated事件并且我的匿名处理程序完成后返回吗?

using forms = System.Windows.Forms;
namespace ProxyProvider
{
    public delegate string finished();
    public static class ProxyProvider
    {
        public static string GetProxy()
        {
            string returnValue = "";
            finished meFinished = () =>
            {
                return returnValue;
            };
            forms.WebBrowser browser = new forms.WebBrowser();
            try
            {
                browser.Navigated += (s, e) =>
                {
                    if (browser.ReadyState == forms.WebBrowserReadyState.Loading) //waiting for browser finishes loading page
                        return;
                    returnValue = ParseHtml(browser.DocumentText); 
                    meFinished();  //I want this function to return only here. After it finishes parsing
                };
                browser.Navigate("http://example.com/proxy-list/");
                return returnValue;  // don't want to return it here
            }
            catch (Exception e)
            {
                forms.MessageBox.Show("ProxyProvider.GetProxy(): " + e.Message);
                return "";
            }
        }
}
}

1 个答案:

答案 0 :(得分:3)

你可以,但是你必须调整一些事情。

这里的基本概念是您的主要方法将阻塞,直到从事件处理程序设置了值。由于您目前处于UI线程,因此您希望异步完成此操作(因此您的窗口仍然保持响应)。

首先我们需要更改方法的签名:

public static Task<string> GetProxy()

我们将在此处返回一个Task对象,以便调用者可以等待我们并获取字符串返回值。然后,您需要设置TaskCompletionSource并在处理程序中设置它:

TaskCompletionSource<string> navTaskSource= new TaksCompletionSource<string>();
browser.Navigated += (s, e) =>
{
     if (browser.ReadyState == forms.WebBrowserReadyState.Loading)
          return;
     string returnValue = ParseHtml(browser.DocumentText); 
     navTaskSource.TrySetResult(returnValue);
};

现在你已经完成了,只需返回任务:

browser.Navigate("http://example.com/proxy-list/");
return navTaskSource.Task;

您的调用代码应await 方法,以避免任何UI线程阻塞并获取返回值。

string proxy = await MyClass.GetProxy();

请注意,包含await的方法必须标记为async