多线程 Delphi 7 App - 应用程序终止问题

Posted

技术标签:

【中文标题】多线程 Delphi 7 App - 应用程序终止问题【英文标题】:Multithreaded Delphi 7 App - a problem with app termination 【发布时间】:2010-12-17 20:25:58 【问题描述】:

我有一个 TThread 的后代和一个对象列表,每个对象都有自己的此类线程的副本,以及使用 CreateEvent() API 创建的 Event 对象。

不同的对象通过事件触发相互交互。 IE。每个线程必须等到某个其他线程触发其事件。当然,有一个“主”线程,它永久工作,因此永远不会发生自阻塞。这个系统在每个对象的 Execute 方法结束前都可以正常工作。

当我尝试中断所有线程时出现问题,例如通过应用关闭。在这种情况下,我需要一些调用每个线程的 Terminate 方法的外部函数:

  for i := 0 to FLayers.Count - 1 do
  begin
    FLayers.Layer[i].FTerminating := true;
    f := true;
    while f do
    begin
      f := FLayers.Layer[i].IsActive;
      if f then
      begin
        Sleep(100);
        Application.ProcessMessages;
      end;
    end;
    FLayers.Layer[i].FTerminating := false;
  end;

此函数位于 Form.OnClose() 事件中。

问题是大约有两个线程正常终止,但其他所有线程都在 WaitForSingleObject() 调用中停止:

procedure TLayerThread.Execute;
begin
FLayer.FIsActive := true;
...............
repeat
 //
 if Terminated or
   FLayer.FTerminating or
   (FLayer.FEvent = INVALID_HANDLE_VALUE) then
   begin
     break;
   end;
 //
 Fres := WaitForSingleObject(FLayer.FEvent, 100); <<<<<<<<<<<<<<<<<<<<<<<<
until Fres <> WAIT_TIMEOUT;
...........
FLayer.FIsActive := false;
end;

所有线程都只是停止(挂起)在线。上面标记,尽管设置了超时值。

有什么想法吗?

我正在使用 Delphi 7 和 Win XP。

提前致谢。

跟进--

我发现问题在 Execute() 方法中的 Synchronize() 调用中得到解决。我不明白这里有什么问题。 Synchronize() 调用通常的东西,比如视觉控件更新等等。

正如调试器所示,我的线程中没有一个挂在某个 WaitForSingleObject() 调用上,但这不是我在 Execute() 方法中用来协调不同线程的方法,而是另一个调用。我可以假设它在这里:

class procedure TThread.Synchronize(ASyncRec: PSynchronizeRecord);
.................
        LeaveCriticalSection(ThreadLock);
        try
          WaitForSingleObject(SyncProc.Signal, INFINITE);<<<<<<<<<<<<<<<<<<<<<<
        finally
          EnterCriticalSection(ThreadLock);
        end;
..................

有没有人可以告诉我我的代码有什么问题?我从来没有听说过不允许在 Execute() 方法中调用 Synchronize()...

【问题讨论】:

【参考方案1】:

您应该使用 WaitForMultipleObjects 而不是 WaitForSingleObject 无限超时并等待两个事件,您的 FLayer.FEvent 和第二个终止事件。

AFAIR 你必须为每个进程创建一个终止事件。 如果 WaitForMultipleObjects 返回终止事件的 ID,则退出循环。

在 OnClose() 方法中,您必须简单地通知所有终止事件。

【讨论】:

很抱歉,您是关于进程终止事件的吗?请注意,我不是要终止进程。让我们将其视为一个中断所有线程的过程。例如。我有“停止处理”菜单项,它不会关闭应用程序,而只是中断所有线程,就像上面显示的那样。而且我不能无限等待事件,因为通常每个线程都必须可以单独中断。简而言之,我的问题是“为什么一个线程终止会导致另一个线程挂起?”谢谢。 在 OnClose() 或其他方法中调用它并不重要。通过为每个线程使用一个关闭事件,您还可以单独结束每个线程。

以上是关于多线程 Delphi 7 App - 应用程序终止问题的主要内容,如果未能解决你的问题,请参考以下文章

delphi 的多线程问题

Delphi多线程怎么处理?

使用delphi多线程实现绑定某个EXE的进程并且向它发送键盘鼠标消息的实例。

delphi 如何在自动终止线程和手动终止线程时都正确的设置窗体上的控件

delphi 没有handle时如何强行终止线程?

delphi 在线程A中终止线程B