4

我想要一个在后台线程中运行的 webrequest,然后在 UI 线程中处理回调。这是我的代码:

Task.Factory.StartNew(() =>
    {
        Console.Out.WriteLine("Start the request");
        return request.GetResponse();
    }
    ).ContinueWith(t =>
    {
        Console.Out.WriteLine("Response");
        using (HttpWebResponse response = (HttpWebResponse)t.Result)
        {
            string responseBody = ...
        }
    },
    TaskScheduler.FromCurrentSynchronizationContext());

    Console.Out.WriteLine("Continue with normal UI code");
}

应该发生什么:

"Start the request"
"Continue with normal UI code"
.
. 
.
"Response"

但我得到:

"Start the request"
.
.
.
"Response"
"Continue with normal UI code"

这就像request.GetResponse()停止独立于 StartNew 的整个线程。有任何想法吗?我正在使用 MonoDevelop(Android 的 Mono)。

编辑:根据Mono for Android 文档

“使用并行任务库创建的任务可以异步运行并返回其调用线程,这使得它们对于触发长时间运行的操作而不会阻塞用户界面非常有用。

一个简单的并行任务操作可能如下所示:"

using System.Threading.Tasks;
void MainThreadMethod ()
{
    Task.Factory.StartNew (() => wc.DownloadString ("http://...")).ContinueWith (
        t => label.Text = t.Result, TaskScheduler.FromCurrentSynchronizationContext()
    );
}

所以我的解决方案应该有效!

如果有人有更好的方式进行后台调用+ UI-thread-callback,我愿意接受建议。尝试了 BeginGetResponse / EndGetResponse 但回调不在 UI 线程上运行。

4

1 回答 1

3

Task.Factory.StartNew()本身不会启动新线程,而是允许在处理当前方法调用时再次调用方法入口点而不会被阻塞。这在幕后是如何发生的有点复杂,但使用它并不是“启动新线程并运行此代码”的方法。最重要的是,在退出调用线程之前,cpu 没有理由不决定在线程中运行代码。这取决于线程调度程序。

于 2013-02-20T16:21:01.887 回答