3

我有一个导出 DLL 并具有library标题的 Delphi 2010 应用程序。它在 TThread 中创建 MainForm,如下所示:

var
  ActiveThread: TActive;
  
type
  TActive= class(TThread)
  protected
    procedure Execute; override;
  end;

procedure TActive.Execute;
begin
      Application.Initialize;
      Application.CreateForm(MyForm, form);
      Application.Run;
end;

begin
  ActiveThread := TActive.Create(true);
  ActiveThread.FreeOnTerminate := true;
  ActiveThread.Resume;
end.

每当我通过该LoadLibrary函数加载此 DLL 时,应用程序运行良好。(显然它使用我传递给LoadLibrary的线程作为主线程并且没有问题)

但是,如果我尝试通过更改选项 -> 应用程序中生成的输出来将此 DLL 导出到实际的 EXE。并将标头从 更改为library然后program构建它并执行输出 EXE 而不是通过 windows api 加载 DLL,应用程序在尝试创建表单时挂起,特别是在Application.CreateForm(MyForm, form);. 如果我从线程中删除应用程序初始化并将其放在主程序上,它运行得很好。

我试图呈现的表单只是一个空表单。有任何想法吗?

4

1 回答 1

4

当将此代码编译为program时,在运行时它将尝试在end.到达时自行终止,甚至在工作线程有机会运行之前,这可能(并且很可能)在Application对象被销毁之后发生。在让程序退出之前,您必须等待工作线程完成其工作,例如:

program MyProgram;

uses
  Classes, Forms, MyForm;

type
  TActive = class(TThread)
  protected
    procedure Execute; override;
  end;

procedure TActive.Execute;
begin
  Application.Initialize;
  Application.CreateForm(TMyForm, MyForm);
  Application.Run;
end;

var
  ActiveThread: TActive;
begin
  ActiveThread := TActive.Create(False);
  ActiveThread.WaitFor;
  ActiveThread.Free;
end.

但是,真的没有充分的理由使用这样的工作线程,这违背了使用线程的全部目的,所以你不妨干脆完全摆脱它:

program MyProgram;

uses
  Forms, MyForm;

begin
  Application.Initialize;
  Application.CreateForm(TMyForm, MyForm);
  Application.Run;
end.

另一方面,如果您想在项目之间共享公共代码programlibrary那么您可以将Application代码包装在一个函数中,让项目决定哪个线程调用该函数,例如:

unit MyApp;

interface

procedure RunMyApp;

implementation

uses
  Forms, MyForm;

procedure RunMyApp;
begin
  Application.Initialize;
  Application.CreateForm(TMyForm, MyForm);
  Application.Run;
end;

end.
program MyProgram;

uses
  MyApp;

begin
  RunMyApp;
end.
library MyLibrary

uses
  Classes, MyApp;
  
type
  TActive = class(TThread)
  protected
    procedure Execute; override;
  end;

procedure TActive.Execute;
begin
  RunMyApp;
end;

var
  ActiveThread: TActive;
begin
  ActiveThread := TActive.Create(True);
  ActiveThread.FreeOnTerminate := True;
  ActiveThread.Resume;
end.
于 2021-02-17T23:28:21.317 回答