在Thread内部使用Timer - IsAlive属性为false

时间:2012-07-06 11:20:30

标签: c# multithreading

您可能知道Thread类具有IsAlive属性。如果返回线程方法或线程被中止,则为false。所以我有一个问题:

我有一个Windows服务,可以在不同的线程中启动多个任务。它会永久检查线程的IsAlive属性,如果为false,则重新创建任务:

    foreach (var worker in _workers)
        _threads.Add(new Thread(worker.ProtectedRun));

    foreach (var thread in _threads)
        thread.Start();

    while (!EventWaitHandle.WaitOne(0))
    {
        for (var i = 0; i < _threads.Count; i++)
        {
            if (!_threads[i].IsAlive)
            {
                _threads[i] = new Thread(_workers[i].ProtectedRun);
                _threads[i].Start();
            }
        }

        EventWaitHandle.WaitOne(1000);
    }

但是其中一个任务有Timer内部。在ProtectedRun方法中,它启动计时器并返回。返回方法后 - &gt; Thread的IsAlive属性变为false - &gt; Windows服务再次启动线程 - &gt;无限循环:)

    public override void ProtectedRun()
    {
        _timer = new System.Timers.Timer(24 * 60 * 1000);
        _timer.Elapsed += OnTimedEvent;
        _timer.Enabled = true;
        _timer.Start();
    }

有人知道如何处理这种情况吗?也许检查线程状态而不是IsAlive属性?

2 个答案:

答案 0 :(得分:3)

嗯,你的设计很糟糕,就像那样。

  1. 您在线程中运行任务,并在它们不再存在时重新启动它们。好。

  2. 您有一个特定的任务立即结束,但在计时器上排队更多的工作。好。

  3. 然而,坏消息是,第一个前提下的一项任务和测试是完全不兼容的。

    选择:

    • 重做重启逻辑或

    • 启动计时器后保持线程活动。

    • 重做它,以便有一个特定的TASK LEVEL(非线程级别)状态字段。

    最后,您的一个ProtectedRun方法在执行排队活动(通过计时器)时返回时不遵循应该遵循的规范。

答案 1 :(得分:1)

发生的事情是您的线程(带有计时器的线程)实际上已成功退出。这就是为什么IsAlive是错误的。

计时器处理程序中执行的代码不会在您创建的主题中执行

请参阅以下示例:

internal class Program
{
    private static void Main(string[] args)
    {
        Thread z = new Thread(new ParameterizedThreadStart(Test)) { Name = "My new thread !" };
        z.Start();

        Console.ReadKey();
    }

    private static void Test(object obj)
    {
        Console.WriteLine(string.Format("My name is: {0}, my ID is: {1}", Thread.CurrentThread.Name, Thread.CurrentThread.ManagedThreadId));

        var _timer = new System.Timers.Timer(1000);
        _timer.Elapsed += (sender, e) =>
            {
                Console.WriteLine(string.Format("[Timer] My name is: {0}, my ID is: {1}", Thread.CurrentThread.Name, Thread.CurrentThread.ManagedThreadId));
            };
        _timer.Enabled = true;
        _timer.Start();
    }
}

结果:

My name is: My new thread !, my ID is: 10
[Timer] My name is: , my ID is: 12
[Timer] My name is: , my ID is: 12
[Timer] My name is: , my ID is: 12

您必须重新考虑应用程序的设计。你确定你需要首先启动一个产生计时器的线程吗?