Как подключить метод CancellationTokenSource к DownloadStringTaskAsync и отменить асинхронный вызов?

Я создаю пример для вызова ссылки с использованием WebClient, используя метод async и wait. Теперь я хочу также добавить функцию отмены асинхронного вызова. Но я не могу получить токен CancellationTokenSource и подключить DownloadStringTaskAsync к этому токену отмены. Следующий Мой код может кто-нибудь сказать мне, как это сделать.

private async void DoWork()
        {
            this.Cursor = Cursors.WaitCursor;
            Write("DoWork started.");
            cts = new CancellationTokenSource();
            WebClient wc = new WebClient();
            string result = await wc.DownloadStringTaskAsync(new Uri("http://gyorgybalassy.wordpress.com"));

            if (result.Length < 100000)
            {
                Write("The result is too small, download started from second URL.");
                result = await wc.DownloadStringTaskAsync(new Uri("https://www.facebook.com/balassy"));
            }
            Write("Download completed. Downloaded bytes: " + result.Length.ToString());
            Write("DoWork ended.");
            this.Cursor = Cursors.Default;
        }

        private void btnCancel_Click(object sender, EventArgs e)
        {
            Write("Cancellation started.");
            this.cts.Cancel();
            Write("Cancellation ended.");
        }

Когда моя кнопка "Отменить" вызывает cts.Cancel, вызов DownloadStringTaskAsync не отменяется. Почему кнопка отмены не может отменить асинхронные вызовы?

Ответ 1

Асинхронные возможности WebClient predate.Net 4.5, поэтому он поддерживает Асинхронный шаблон на основе задач только частично. Это включает в себя собственный механизм отмены: метод CancelAsync(), который работает даже с новыми методами -TaskAsync. Чтобы вызвать этот метод, когда CancellationToken отменен, вы можете использовать его метод Register():

cts.Token.Register(wc.CancelAsync);

В качестве альтернативы вы можете использовать новый HttpClient, как предложил Стивен, который полностью поддерживает TAP, включая CancellationToken s.

Ответ 2

WebClient не поддерживает отмену. Я рекомендую использовать более новый тип, например HttpClient:

...
cts = new CancellationTokenSource();
string result;
using (var client = new HttpClient())
using (var response = await client.GetAsync("http://gyorgybalassy.wordpress.com", cts.Token))
{
  result = await response.Content.ReadAsStringAsync();
}

if (result.Length < 100000)
...

Метод GetAsync по умолчанию не будет завершен, пока он не прочитает весь ответ, поэтому строка await response.Content.ReadAsStringAsync будет фактически завершена синхронно.

Ответ 3

Методы расширения, основанные на ответе svick:

public static async Task<string> DownloadStringTaskAsync(this WebClient webClient, string url, CancellationToken cancellationToken) {
    using (cancellationToken.Register(webClient.CancelAsync)) {
        return await webClient.DownloadStringTaskAsync(url);
    }
}

public static async Task<string> DownloadStringTaskAsync(this WebClient webClient, Uri uri, CancellationToken cancellationToken) {
    using (cancellationToken.Register(webClient.CancelAsync)) {
        return await webClient.DownloadStringTaskAsync(uri);
    }
}