当我在控制台窗口中使用滚动条时,所有线程和任务都将暂停。有什么方法可以防止这种情况发生吗?
例子:
Task.Run(async () =>
{
for (var z = 0; true; z++)
{
Console.WriteLine(z);
await Task.Delay(200);
}
});
或者
var s = new Thread(() =>
{
for (var z = 0; true; z++)
{
Console.WriteLine(z);
Thread.Sleep(200);
}
});
如果您将这些代码示例之一作为控制台应用程序运行,则两者将执行相同的操作:打印一些数字。在执行过程中,用鼠标左键按住滚动条一段时间,数字将停止显示。任务/线程已暂停。释放鼠标按钮时,数字将继续。
因为释放滚动条后的数字跟在前一个数字之后,所以控制台并没有简单地停止数字的输出。它一直在Console.WriteLine-方法中等待,直到释放滚动条为止。
有人知道如何解决我的问题吗?
您可以采取多种措施来阻止这种情况。Console.WriteLine
在您按住滚动条时将阻塞,这与线程或任务完全无关。下面的小程序显示了相同的问题。
using System;
using System.Threading;
namespace ConsoleApplication1
{
class Program
{
static void Main(string[] args)
{
for (var z = 0; true; z++)
{
Console.WriteLine(z);
Thread.Sleep(200);
}
}
}
}
您不能更改行为Console.WriteLine
的唯一方法就是在缓冲区和控制台写入之间放置一个缓冲区。
using System;
using System.Collections.Concurrent;
using System.Threading;
namespace ConsoleApplication1
{
class Program
{
static void Main(string[] args)
{
//The writer lives in another thread.
var s = new Thread(ConsoleWriter);
s.Start();
for (var z = 0; true; z++)
{
//This add call does not get blocked when Console.WriteLine gets blocked.
LinesToWrite.Add(z.ToString());
Thread.Sleep(200);
}
}
static void ConsoleWriter()
{
foreach (var line in LinesToWrite.GetConsumingEnumerable())
{
//This will get blocked while you hold down the scroll bar but
// when you let go it will quickly catch up.
Console.WriteLine(line);
}
}
static readonly BlockingCollection<string> LinesToWrite = new BlockingCollection<string>();
}
}
这只是一个快速而肮脏的解决方案,还有其他解决方法可以使其更加透明,例如使派生自该类的类TextWriter
将缓冲调用,并在对的调用中使用该类Console.SetOut(
。
本文收集自互联网,转载请注明来源。
如有侵权,请联系 [email protected] 删除。
我来说两句