Как подключить метод 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);
}
}