Почему код, приведенный ниже, не заполняется, если вы не вводите какой-либо ввод, и почему он все еще реагирует на нажатие клавиши даже после отмены маркера отмены?Как отменить Stream.ReadAsync?
// Set up a cancellation token
var cancellationSource = new CancellationTokenSource();
// Cancel the cancellation token after a little bit of time
Task.Run(async() =>
{
await Task.Delay(TimeSpan.FromSeconds(2));
cancellationSource.Cancel();
Console.WriteLine("Canceled the cancellation token");
});
// Wait for user input, or the cancellation token
Task.Run(async() =>
{
try
{
using (var input = Console.OpenStandardInput())
{
var buffer = new byte[1];
Console.WriteLine("Waiting for input");
await input.ReadAsync(buffer, 0, 1, cancellationSource.Token); // This is impossible to cancel???
Console.WriteLine("Done waiting for input"); // This never happens until you press a key, regardless of the cancellation token
}
}
catch (Exception e)
{
Console.WriteLine(e.Message); // No errors
}
})
.Wait(); // Block until complete
The documentation for Stream.ReadAsync says:
Если операция будет отменена до ее завершения, возвращаемая задача содержит Отменено значения для свойства Status.
Это означает, что отмена маркера отмены отменяет операцию, верно? Тем не менее, по какой-то причине the source code for Stream.ReadAsync ничего с маркером отмены не делать, если он не отменен заранее:
public virtual Task<int> ReadAsync(Byte[] buffer, int offset, int count, CancellationToken cancellationToken)
{
// If cancellation was requested, bail early with an already completed task.
// Otherwise, return a task that represents the Begin/End methods.
return cancellationToken.IsCancellationRequested
? Task.FromCancellation<int>(cancellationToken)
: BeginEndReadAsync(buffer, offset, count);
}
Поэтому лексема параметр отмены бессмысленно - как я могу отменить, что асинхронная читать?
Обратите внимание, что Console.OpenStandardInput возвращается экземпляр [__ConsoleStream] (https://referencesource.microsoft.com/#mscorlib/system/io/__consolestream.cs,de9f3a925342686c), который не переопределяет .ReadAsync –
В следующий раз прежде чем вы сделаете ненужную магию, чтобы отменить токен через некоторое время, используйте соответствующий конструктор: 'var cancelationSource = new CancellationTokenSource (TimeSpan.FromSeconds (2));'. https://msdn.microsoft.com/en-us/library/hh139229(v=vs.110).aspx ' –
@PeterBons Для меня мой способ более с готовностью сообщает, что что-то будет распечатано на консоли после –