您可能知道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属性?
答案 0 :(得分: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
您必须重新考虑应用程序的设计。你确定你需要首先启动一个产生计时器的线程吗?