如何将 C++ 联合转换为 Delphi

How to convert a C++ union to Delphi

我知道之前有人问过这个问题,但我尝试将一些 C++ structures/unions 转换为 Delphi 以使用 Hikvision SDK 失败。

我尝试转换的C++ structures/unions如下:

struct{
  BYTE                           byEnable;
  BYTE                           byRes1[3];
  DWORD                          dwTriggerType;
  NET_ITC_TRIGGER_PARAM_UNION    uTriggerParam;
  BYTE                           byRes[64];
}NET_ITC_SINGLE_TRIGGERCFG,*LPNET_ITC_SINGLE_TRIGGERCFG;

union{
  DWORD                              uLen[1070];
  NET_ITC_POST_IOSPEED_PARAM         struIOSpeed;
  NET_ITC_POST_SINGLEIO_PARAM        struSingleIO;
  NET_ITC_POST_RS485_PARAM           struPostRs485;
  NET_ITC_POST_RS485_RADAR_PARAM     struPostRadar;
  NET_ITC_POST_VTCOIL_PARAM          struVtCoil;
  NET_ITC_EPOLICE_IOTL_PARAM         struIOTL;
  NET_ITC_EPOLICE_RS485_PARAM        struEpoliceRs485;
  NET_ITC_EPOLICE_RS485_PARAM        struPERs485;
}NET_ITC_TRIGGER_PARAM_UNION,*LPNET_ITC_TRIGGER_PARAM_UNION;

我试过以下方法:

  PNetItcSingleTriggerCfg = ^TNetItcSingleTriggerCfg;
  TNetItcSingleTriggerCfg = record
    byEnable:      Byte;
    byRes1:        array [0..2] of Byte;
    dwTriggerType: DWord;
    uTriggerParam: TNetItcTriggerParamUnion;
    byRes:         array [0..63] of Byte;
  end;

  PNetItcTriggerParamUnion = ^TNetItcTriggerParamUnion;
  TNetItcTriggerParamUnion = record
    case integer of
      0: (uLen:             array [0..1069] of DWord);
      1: (struIOSpeed:      TNetItcPostIOSpeedParam);
      2: (struSingleIO:     TNetItcPostSingleIOParam);
      3: (struPostRs485:    TNetItcPostRS485Param);
      4: (struPostRadar:    TNetItcPostRS485RadarParam);
      5: (struVtCoil:       TNetItcPostVTCoilParam);
      6: (struHvt:          TNetItcPostHvtParam);
      7: (struIOTL:         TNetItcEPoliceIOTLParam);
      8: (struEpoliceRs485: TNetItcEPoliceRS485Param);
      9: (struPERs485:      TNetItcEPoliceRS485Param);          
      10:(struPostMpr:      TNetItcPostMprParam);
      11:(struViaVtCoil:    TNetDvrViaVtCoilParam);
      12:(struPostImt:      TNetItcPostImtParam);
      13:(struPostPrs:      TNetItcPostPrsParam);
      14:(struIpcHvt:       TNetIpcPostHvtParam);
      15:(struHvtV50:       TNetIpcPostHvtParamV50);
  end;

而且我也尝试过将其作为嵌套记录(如此处建议http://rvelthuis.de/articles/articles-convert.html#unions

  PNetItcSingleTriggerCfg = ^TNetItcSingleTriggerCfg;
  TNetItcSingleTriggerCfg = record
    byEnable:      Byte;
    byRes1:        array [0..2] of Byte;
    dwTriggerType: DWord;
    uTriggerParam: record
      case integer of
        0: (uLen:             array [0..1069] of DWord);
        1: (struIOSpeed:      TNetItcPostIOSpeedParam);
        2: (struSingleIO:     TNetItcPostSingleIOParam);
        3: (struPostRs485:    TNetItcPostRS485Param);
        4: (struPostRadar:    TNetItcPostRS485RadarParam);
        5: (struVtCoil:       TNetItcPostVTCoilParam);
        6: (struHvt:          TNetItcPostHvtParam);
        7: (struIOTL:         TNetItcSingleIOTLParam);
        8: (struEpoliceRs485: TNetItcEPoliceRS485Param); 
        9: (struPERs485:      TNetItcEPoliceRS485Param);
        10:(struPostMpr:      TNetItcPostMprParam);
        11:(struViaVtCoil:    TNetDvrViaVtCoilParam);
        12:(struPostImt:      TNetItcPostImtParam);
        13:(struPostPrs:      TNetItcPostPrsParam);
        14:(struIpcHvt:       TNetIpcPostHvtParam);
        15:(struHvtV50:       TNetIpcPostHvtParamV50);
      end;
    byRes:         array [0..63] of Byte;
  end;

我在这里看到了类似的问题(即 How do I translate a C union into Delphi? ),但我的示例中的联合位于结构的中间,据我所知,case 语句的“结束”也结束了记录.

我想我理解变体记录背后的理论,该变体记录对 case 语句中的字段具有相同的内存分配,因此使用的字段将是 either/or,但我无法解决的是如何告诉 DLL 如何访问这些记录,是 struName.unionName.fieldName 还是 struName.fieldName 以及联合是如何定义的(即 case 语句的选择器是什么,以及如何知道什么数据类型选择器是)。

我有三个类似的结构要翻译,我想如果我能破解一个,我就能破解它们。

按照上述记录,我一直收到错误信息‘参数错误。从 DLL 调用函数时,SDK API 中的输入或输出参数为 NULL(如果您需要这方面的更多信息,请询问),这让我认为我的记录未正确转换。

我正在使用 HCNetSDK.dll SDK 版本 5.0.3.20,如果有帮助,我的 IDE 是 XE7。

如有任何帮助,我们将不胜感激。

假设结构对齐设置匹配,那么您转换联合的两次尝试都是正确的。您可以使用任何您喜欢的。对于它的价值,我更喜欢第一种方法,你声明一个类型来表示联合。

无论您的实际问题是什么,它似乎都不在于联合转换。一个简单的测试方法是检查 C++ 和 Delphi 版本中类型的大小是否匹配,以及每个成员的偏移量是否匹配。

要测试类型的大小,在 C++ 中使用 sizeof,在 Delphi 中使用 SizeOf。对于偏移量,使用 offsetof macro for C++ and the trick shown in my answer here for Delphi.