【发布时间】:2012-04-30 02:15:29
【问题描述】:
希望我能体面地解释这个问题!
我正在尝试将 ReWire 音频设备实现为 Delphi .dll。如果您不知道 ReWire 是什么,请不要担心。重要的是我的代码编译成 .dll,我从 ReWire 系统调用我的 .dll 来打开显示器,检查显示器是否打开,然后再次关闭它。
当我接到启动电话时,我会执行以下操作:
if not Assigned(form) then
form := TMyForm.Create(nil);
form.Show;
form 是我的 Delphi 库中的一个全局变量(可能有问题?)。我已经连接了MyForm 的OnCreate 事件来做一些有趣的事情,比如准备一系列我想要使用的东西。
到目前为止一切都很好。我的表单中有一个小按钮,可以打开TOpenDialog。我发现一旦该对话框关闭,OnCreate 事件就会在我的表单中再次触发!
我检查了OnDestroy 没有被调用,所以我不知道为什么OnCreate 会再次被调用。
不幸的是,我不确定哪些信息是相关的,但这是第一次调用堆栈(首次设置表单时):
正如预期的那样,ReWire 正在调用我的 .dll 以启动 Panel 应用程序,因此我创建了我的表单。太好了,情况看起来不错。
然后在我的表单中,我打开一个小对话框,选择一个文件,然后执行一些操作。在左侧字段之外,再次调用OnCreate,这是当时的调用堆栈:
这是一个疯狂的电话聚会! Reaper(在底部)是我用来测试我的应用程序的 ReWire 主机,但我不知道堆栈跟踪内部发生了什么,因为它都不是我的代码。突然间,我认为不应该调用该事件,因为 OnDestroy 甚至没有被调用。
我能想到的唯一另一件重要的事情是,如果我打印出Sender 的地址,它每次都不同,所以它会以某种方式再次被创建,但我'已经检查过我只调用了 MyForm.Create 一次。
关于这种事情如何发生的任何想法?
【问题讨论】:
-
这行不通。你有两个 VCL 实例。一个在您的 exe 中,一个在 DLL 中。将其全部静态链接到一个 exe 中。或者使用包。
-
从放弃 OnCreate 逻辑开始,看看你是否可以让它工作。然后,将逻辑移至发布在 OnShow 处理程序上的 WM_USER 消息处理程序上的新函数。这样用户就可以尽早看到表单,并且可以看到正在发生的设置(没有延迟)。这将允许您将两个部分分开,其中一个错误将更容易识别。
-
非常感谢大家的帮助!答案类似于@Remy Lebeau 的建议,并在他的回复下方的评论中详细说明。我真的很感谢大家的意见!我有点担心 David Heffernan 在那里的评论——我对所有这些 VCL 东西都是新手,所以如果其他事情变得疯狂,我会确保考虑如何将 VCL 实例保持为一个!