取消异步检索URL的任务

时间:2012-01-31 04:34:07

标签: c# asynchronous multitasking task

我在查找如何在C#中取消此任务时遇到了一些问题。我没有完全理解处理线程,我已经尝试使用谷歌搜索一些简单的代码示例来帮助我,但我真的没有在哪里。以下是我正在处理的代码:

var tasks = urls.Select(url => Task.Factory.StartNew(
state =>
{
    using (var client = new WebClient())
    {

        lock (this)
        {

        // code to download stuff from URL

        }

    }
}, url)).ToArray();

    try
    {
       Task.WaitAll(tasks);
    }
    catch (Exception e)
    {
      textBox2.AppendText("Error: " + e.ToString());
    }

哪里"网址"是一个URL数组。是否有一种简单的方法可以使我在单击程序中的按钮时完全停止下载URL?另外,我粘贴的代码片段在backgroundWorker1调用的函数中,我想这可能会使事情变得复杂一些。 (我之所以拥有backgroundWorker,是因为用户界面在下载网址时没有锁定。)

如果这有点令人困惑,这里是我试图用我的代码实现的概述:

  • 我有一系列网址,我想在不锁定用户界面的情况下异步下载每个网址。
  • 我最喜欢用户通过点击按钮停止程序下载网址,几乎取消了该帖子。
  • 当用户再次单击该按钮时,程序将从该阵列再次下载URL。

提前致谢。

2 个答案:

答案 0 :(得分:1)

使用WebClient,您可以使用CancelAsync方法取消异步操作。

要取消您通过Factory.StartNew启动的任务,您应该使用CancellationTokenSource。您需要将CancellationTokenSource.Token传递给任务(并且您可以使用token.IsCancellationRequested询问是否已取消令牌),并且您要调用CancellationTokenSource.Cancel()将令牌设置为已取消。

答案 1 :(得分:1)

不知道这是否是正确的方法,但我已经能够使用以下代码取消任务。我创建了一个包含ListBoxProgressBar的表单,因此我正在筹集并处理ProgressChanged BackgroundWorker事件。希望这能以某种方式帮助你。

void bw_DoWork(object sender, DoWorkEventArgs e)
{
    CancellationTokenSource _tokenSource = new CancellationTokenSource();
    CancellationToken _token = _tokenSource.Token;

    var urls = e.Argument as IEnumerable<string>;

    _token = new CancellationToken();

    if (urls == null) return;

    var i = 0;
    var a = 100 / urls.Count();

    var sb = new StringBuilder();
    var t = urls.Select(url => Task.Factory.StartNew(
                    (u) =>{
                        using (var wc = new WebClient())
                        {
                            lock (this){
                                var s = wc.DownloadString(u.ToString());
                                sb.AppendFormat("{1}:{0}\r\n", "", u);
                            }
                        }

                    if (Worker.CancellationPending){
                        //MAGIC HAPPENS HERE, IF BackgroundWorker IS REQUESTED
                        //TO CANCEL, WE CANCEL CancellationTokenSource
                        _tokenSource.Cancel();
                    }

                    //IF CANCELATION REQUESTED VIA CancellationTokenSource
                    //THROW EXCEPTION WHICH WILL ADD TO AggreegateException
                    _token.ThrowIfCancellationRequested();

                    //YOU CAN IGNORE FOLLOWING 2 LINES
                    i += a;
                    Worker.ReportProgress(i, u);
    }, url, _token)).ToArray();

    try
    {
        Task.WaitAll(t);
    }
    catch (AggregateException age)
    {
        if (age.InnerException is OperationCanceledException)
            sb.Append("Task canceled");
    }
    catch (Exception ex)
    {
        sb.Append(ex.Message);
    }

    e.Result = sb;
}