1

我需要在我的 delphi XE3 应用程序中使用 DLL,我收到了一个用 c++ 编写的演示应用程序,它显示了如何调用 DLL。
我已经成功调用了 DLL 和 dll 的 Initialize 方法,但是我没有从 DLL 得到任何消息。
这是 C++ 源代码:
创建消息处理程序:

#define WM_POSSTATE  WM_APP+1
#define ON_WM_POSSTATE() \
{ WM_POSSTATE, 0, 0, 0, AfxSig_vwl, \
    (AFX_PMSG)(AFX_PMSGW)(void (AFX_MSG_CALL CWnd::*)(UINT, LPARAM))&OnPOS },

Initialize 方法的定义:

typedef int (INITIALIZE)( char * cPort , UINT Msg, HWND *hWnd_p );
INITIALIZE *Initialize;

HMODULE         hPosDll;

加载 DLL:

if( (hPosDll = LoadLibrary( "posdll.dll" ) ) == NULL )
{
    MessageBox( "Error: can not open posdll.dll",NULL,MB_OK);
    exit(1);
}

if( (::Initialize = (INITIALIZE*)GetProcAddress( hPosDll, "Initialize" )) == NULL )
{
    MessageBox( "Error: can not find function",NULL,MB_OK);
    FreeLibrary( hPosDll );
    exit(1);
}

调用初始化方法:

res=::Initialize('com3', WM_POSSTATE , &m_hWnd);

if( res )
{
    MessageBox( "Error opening comms",NULL,MB_OK);
    FreeLibrary( hPosDll );
    exit(1);
}

在 Initialize 调用之后,从 DLL 到应用程序的消息由 OnPos 方法处理:

void CPosDemoDlg::OnPOS(UINT Result, LPARAM Param )
{
    DoStuff;
}

在德尔福我到目前为止:

unit UFrmMain;

interface

uses
  Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,
  Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.StdCtrls, MCPOSDLL;

const
  WM_POSSTATE = WM_APP + 1;
  MCDLL = 'posdll.dll';

type
  TInitialize = function(cport: PAnsiChar; Msg: Integer; Handle: HWND):Integer; stdcall;

  TFrmMain = class(TForm)
    EdCOMPort: TEdit;
    BtnConnect: TButton;
    LblCOMPort: TLabel;
    procedure ON_WM_POSSTATE(var Msg: TMessage); message WM_POSSTATE;
    procedure BtnConnectClick(Sender: TObject);
  private
    DLLHandle: THandle;
  public
    { Public declarations }
  end;

var
  FrmMain: TFrmMain;

implementation

{$R *.dfm}

{-------------------------------------------------------------------------------}
 procedure TFrmMain.BtnConnectClick(Sender: TObject);
 var
   MCInitialize: TInitialize;
   Res: Integer;
 begin
   DLLHandle := LoadLibrary(PChar(MCDLL));

   if DLLHandle <> 0 then
     begin
       @MCInitialize := getProcAddress(DLLHandle, 'Initialize');

       if @MCInitialize <> NIL then
         begin
           Res := MCInitialize(PAnsiChar('com3'), WM_POSSTATE, Self.Handle);

           if Res <> 0 then
             begin
               MessageDlg('Error opening comms', mtWarning, [mbOK], 0);
             end;
         end;
     end
   else
     begin
       MessageDlg('posdll.dll could not be located.', mtWarning, [mbOK], 0);
     end;
 end;

{-------------------------------------------------------------------------------}
 procedure TFrmMain.ON_WM_POSSTATE(var Msg: TMessage);
 begin
   showmessage('Message received');
 end;

end.

使用此解决方案,我可以激活连接到 com3 的设备,但永远不会触发消息处理程序。
我认为&m_hWnd在 C++ 中使用 Initialize 方法发送的内容与Self.Handle我在 Delphi 中发送的内容不同。
谁能帮我吗?提前致谢。

4

3 回答 3

3

C++ 函数声明为:

typedef int (INITIALIZE)( char * cPort , UINT Msg, HWND *hWnd_p );

您的 Delphi 等效项声明为:

TInitialize = function(cport: PAnsiChar; Msg: Integer; 
  Handle: HWND): Integer; stdcall;

我可以在您的 Delphi 声明中看到以下差异:

  1. Msg参数是无符号的,但Delphi 翻译是有符号的。
  2. hWnd_p参数是指向的指针HWND。在 Delphi 中,您按值传递窗口句柄。
  3. C++ 代码的调用约定是cdecl,但您将 Delphi 代码声明为stdcall.

所以在 Delphi 中,声明应该是:

TInitialize = function(cport: PAnsiChar; Msg: Cardinal; 
  Handle: PHWND): Integer; cdecl;

wherePHWND是声明为的类型^HWND。我怀疑这种类型是在Windows单元中声明的,但如果没有,那么你可以很容易地声明它。

或者更简单的翻译可能是:

TInitialize = function(cport: PAnsiChar; Msg: Cardinal; 
  var Handle: HWND): Integer; cdecl;

当然,您在转录 C++ 代码时省略了调用约定是合理的。stdcall即使您在问题中放置的声明没有这样说,也许该功能确实如此。

除此之外,您的代码正在使用表单的窗口句柄。这可能是有风险的。在某些情况下,VCL 会重新创建窗口。这将导致您传递给 DLL 的句柄被破坏。DLL 将继续向该窗口发送消息,但您的表单将停止接收它们,因为它的窗口已更改。

解决这个问题的方法是使用 VCL 不会重新创建的窗口句柄。通过调用获得其中之一AllocateHwnd

打电话时要小心InitializePAnsiChar('com3')您作为端口名称传递。您需要确保文字'com3'确实是 ANSI 编码的。理想情况下,您应该'com3'直接传递给Initialize. 如果那不能编译(我不确定是否会编译)然后通过PAnsiChar(AnsiString('com3')).

于 2013-10-09T10:26:22.357 回答
1

乍一看,它看起来像你的声明

TInitialize = function(cport: PAnsiChar; Msg: Integer; Handle: HWND):Integer;

不匹配

typedef int (INITIALIZE)( char * cPort , UINT Msg, HWND *hWnd_p );

请注意,C 声明需要一个指向持有句柄的变量的指针。

尝试

TInitialize = function(cport: PAnsiChar; Msg: Integer; var Handle: HWND):Integer;

看看你能走多远。

于 2013-10-09T08:32:03.823 回答
0

在 Hanno 和 David 的帮助下,我更改了我的代码,它现在可以工作了。

我所做的更改:

type
  PHWND = ^HWnd;    
  TInitialize = function(cport: PAnsiChar; Msg: Cardinal; Handle: PHWND):Integer; stdcall;

var 
  TmpHandle
begin
  ...
  TmpHandle := Self.Handle;
  Res := MCInitialize(PAnsiChar(AnsiString('com3')), WM_POSSTATE, @TmpHandle);

根据 David Hefferman 的建议,我将在最终应用程序中创建一个单独的组件,这样我将使用这种方式AllocateHwnd创建一个固定句柄。...

于 2013-10-10T07:30:28.493 回答