【问题标题】:Cancelling a DownloadStringTaskAsync and restart it [duplicate]取消 DownloadStringTaskAsync 并重新启动它[重复]
【发布时间】:2014-12-25 14:13:00
【问题描述】:

我正在使用 System.Net.WebClient.DownloadStringTaskAsync 异步从网页下载字符串。

大约 5 次中有 4 次成功完成,但偶尔会挂起,我必须重新启动程序才能“重置”它。

因此我正在考虑实现一个超时功能,它会在 5 秒后自动取消任务并重新启动它:

async void DoTasks()
{
    string output;

    int timeout = 5000;
    WebClient client = new WebClient() { Encoding = Encoding.UTF8 };
    Task<string> task = client.DownloadStringTaskAsync(url);
    if (await Task.WhenAny(task, Task.Delay(timeout)) == task)
    {
        output = task.Result;
    }
    else
    {
        client.CancelAsync();
        DoTasks();
    }
}

然而,这会返回一个NullReferenceException

【问题讨论】:

  • 这段代码不可能导致NullReferenceException。可能它在GetTasks 方法中。此外,您不会处理WebClient
  • 你需要在问题中添加GetTasks方法。
  • 对不起,GetTasks 应该是 DoTasks - 重新启动方法
  • 这里没有明显的可能导致NullReferenceException。哪一行抛出异常?
  • 您还应该发布导致挂起的原始代码。你实际上是在问两个不相关的问题。

标签: c# asynchronous task webclient cancellation


【解决方案1】:

我不是等待/异步方面的专家,我还在学习自己,但也许你应该做这样的事情?

using System;
using System.Net;
using System.Text;
using System.Threading;
using System.Threading.Tasks;

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine("Inline code to test");
            Console.WriteLine(StartTask(true));

            Console.WriteLine("Get actual website to test, write the length of the text only");
            Console.WriteLine(StartTask(false).Length);

            Console.Write("Press any key");
            Console.ReadKey();
        }

        static string StartTask(bool DoTestInsteadOfGetUrl)
        {
            var task = DoTasks(DoTestInsteadOfGetUrl);
            Task.WaitAny(task);

            return task.Result;
        }

        async static Task<string> DoTasks(bool DoTestInsteadOfGetUrl)
        {
            int timeout = 20;
            int count = 1;
            CancellationTokenSource cts;
            string output;                
            do
            {
                cts = new CancellationTokenSource(timeout);
                if (DoTestInsteadOfGetUrl)
                {
                    output = await Task<string>.Run(() => test(count, cts.Token), cts.Token);
                }
                else
                {
                    WebClient client = new WebClient() { Encoding = Encoding.UTF8 };
                    output = await Task<string>.Run(() => client.DownloadStringTaskAsync("http://www.google.com"), cts.Token);
                }

                if (cts.IsCancellationRequested)
                {
                    Console.WriteLine("try #" + count++);
                    timeout += DoTestInsteadOfGetUrl ? 100 : 50;
                }
            } while (cts.IsCancellationRequested);

            return output;
        }

        async static Task<string> test(int count, CancellationToken ct)
        {
            int sleep = 400 - (count * 5);
            await Task.Run(() => Task.Delay(sleep), ct);

            if (!ct.IsCancellationRequested)
            {
                Console.WriteLine("succeed #" + count);
            }
            return sleep.ToString();
        }
    }
}

【讨论】:

  • 这段代码有很多问题。对于初学者,它不包括 OP 的 WebClient 调用,您需要在 DoTasks 中进行调用。但是您的DoTasksvoid,并且不包含async 修饰符。这目前不能解决 OP 的问题。
  • @YuvalItzchakov,我更新了我的代码...希望它更好
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2010-10-12
  • 1970-01-01
  • 2020-06-05
相关资源
最近更新 更多