如何停止控制台暂停我的线程和任务?

演员

当我在控制台窗口中使用滚动条时,所有线程和任务都将暂停。有什么方法可以防止这种情况发生吗?

例子:

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] 删除。

编辑于
0

我来说两句

0 条评论
登录 后参与评论

相关文章