0

这是我用 C# 编写的代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;

namespace xyz
{
    class Program
    {
        Thread t1, t2;
        static void Main(string[] args)
        {
            Program go = new Program();
            go.actual();
        }
        public void actual()
        {
            t1 = new Thread(timer);
            t2 = new Thread(inputting);
            t1.Start();
            t2.Start();
        }
        public void timer()
        {
            Thread.Sleep(5000);
            t2.Abort();
            Console.WriteLine("5 seconds passed!");
        }
        public void inputting()
        {
            Console.WriteLine("Please wait 5 seconds...");
            Console.ReadKey();
        }
    }
}

现在,问题是,当控制台说“5 秒过去了!”时 (中止t2线程后),它不会立即退出。此文本会在那里停留几秒钟,然后控制台退出。

问题是,如果我在线程中止之前按下(ReadKey方法的)一个键,它会显示“5 秒过去了!” 文本,然后它立即退出。

如果我不单击任何键,并且该ReadKey方法没有发生,它只会显示文本几秒钟。

这是为什么?它是一个错误吗?我可以修复它吗?

4

2 回答 2

0

您看到的结果对我来说似乎是合理的,调用Thread.Abort不会以某种方式解除控制台的阻塞,因此它仍会等待输入。无论如何,您应该避免调用Thread.Abort,因为不能保证它会按预期工作。这是在您的线程中构建取消支持的更好方法,并且有多种方法可以做到这一点。

或者,只需使用已经内置的任务并行库。

于 2013-10-11T12:46:44.730 回答
0

从您的评论中,我看到您基本上希望Console.ReadKey在一定时间后使通话超时。不幸的是,ReadKey没有超时参数。好的,可以。我们可以将它包装在一个方法中。您不需要使用任何异步机制来完成此操作。只需Console.KeyAvailable在同步轮询循环中使用。

public static class ConsoleEx
{
  public static bool TryReadKey(TimeSpan timeout, out ConsoleKeyInfo keyinfo)
  {
    var cts = new CancellationTokenSource();
    return TryReadKey(timeout, cts.Token, out keyinfo);
  }

  public static bool TryReadKey(TimeSpan timeout, CancellationToken cancellation, out ConsoleKeyInfo keyinfo)
  {
    keyinfo = new ConsoleKeyInfo();
    DateTime latest = DateTime.UtcNow.Add(timeout);
    do
    {
        cancellation.ThrowIfCancellationRequested();
        if (Console.KeyAvailable)
        {
            keyinfo = Console.ReadKey();
            return true;
        }
        Thread.Sleep(1);
    }
    while (DateTime.UtcNow < latest);
    return false;
  }
}

然后你会像这样使用它。

public static void Main()
{
  ConsoleKeyInfo cki;
  if (ConsoleEx.TryReadKey(TimeSpan.FromSeconds(5), out cki))
  {
    Console.WriteLine("ReadKey returned a value.");
  }
  else
  {
    Console.WriteLine("ReadKey timed out.
  }
}
于 2013-10-11T14:31:22.667 回答