4

我正在将最初用 vb6 编写的应用程序转换为 vb.net。该应用程序所做的其中一件事是将“类型”对象发送到 dll。我尝试将类型转换为结构并 p/调用 dll,但它似乎不起作用。我被困了一个星期,任何帮助将不胜感激

这是该类型的 vb6 代码

'Define WICS Communications Control Block (CCB).
Type WicsCCBType ' Create user-defined type.
    CCBNum As String * 1
    CCBVer As String * 1
    Resp1  As String * 4
    Resp2  As String * 4
    PLUA   As String * 8
    LLUA   As String * 8
    Mode   As String * 8
    ReqMax As String * 5
    ResMax As String * 5
End Type      

这是调用 dll 的方式

Private Declare Sub WICSRASP Lib "wicsrasp.dll" (MyWicsCCB As WicsCCBType)
WICSRASP MyWicsCCB

这是我用 vb.net 尝试过的,但它不起作用

'Define WICS Communications Control Block (CCB).
    <System.Runtime.InteropServices.StructLayoutAttribute( _
            System.Runtime.InteropServices.LayoutKind.Sequential, _
            CharSet:=System.Runtime.InteropServices.CharSet.[Unicode])> _
    Structure WicsCCBType ' Create user-defined type.
        <MarshalAs(UnmanagedType.ByValTStr, sizeconst:=1)> Dim CCBNum As String
        <MarshalAs(UnmanagedType.ByValTStr, sizeconst:=1)> Dim CCBVer As String
        <MarshalAs(UnmanagedType.ByValTStr, sizeconst:=4)> Dim Resp1 As String
        <MarshalAs(UnmanagedType.ByValTStr, sizeconst:=4)> Dim Resp2 As String
        <MarshalAs(UnmanagedType.ByValTStr, sizeconst:=8)> Dim PLUA As String
        <MarshalAs(UnmanagedType.ByValTStr, sizeconst:=8)> Dim LLUA As String
        <MarshalAs(UnmanagedType.ByValTStr, sizeconst:=8)> Dim Mode As String
        <MarshalAs(UnmanagedType.ByValTStr, sizeconst:=5)> Dim ReqMax As String
        <MarshalAs(UnmanagedType.ByValTStr, sizeconst:=5)> Dim ResMax As String
    End Structure

这是我试图称呼它的地方

<System.Runtime.InteropServices.DllImportAttribute("C:\windows\system32\wicsrasp.dll")> _
    Public Shared Sub WICSRASP(
                    ByRef CCB As WicsCCBType,
                    ByRef Request As DAWicsRequestType,
                    ByRef Response As DAWicsResponseType)
    End Sub

 Dim CCB As New modWICSDiary.WicsCCBType()
 CCB.CCBNum = "B"
            CCB.CCBVer = "2"
            CCB.LLUA = "        "
            CCB.Mode = "CICSMO2 "
            CCB.ReqMax = "2100 "
            CCB.ResMax = "2100 "
            CCB.Resp1 = "0   "
            CCB.Resp2 = "0   "
            CCB.PLUA = "WICSPLU "

  NativeMethods.WICSRASP(CCB)

至于值,相同的值适用于 vb6 类型

再次提前谢谢你

4

3 回答 3

1

VB6 会将所有这些字符串元素“编组”为 ANSI 字符串。相应地更改 Vb.Net 编组代码。

  • 试试UnmanagedType.LPStr那些MarshalAs属性?
  • 尝试将 CharSet 从 Unicode 更改为 Ansi?
  • 在? Pack=4_StructLayoutAttribute

有用的链接,解释了 VB6 所做的假设Declare

于 2012-05-31T17:04:57.950 回答
1

我不知道OP是否已经解决了这个问题,但这是我的看法。

VB6 中 UDT 内的固定长度字符串不作为指针编组,而是内联到结构中。此外,VB6 在编组之前将 Unicode 转换为 Ansi。而VB6使用4字节对齐。

该类型将在内存中使用填充看起来这个,每个连续的字段在插图中被命名为 A 到 I,并且由于对齐,_ 是一个填充字节。

CCBNum As String * 1     
|
|+-CCBVer As String * 1     
||
||  Resp1  As String * 4
||  |
||  |   Resp2  As String * 4     
||  |   |
||  |   |   PLUA   As String * 8     
||  |   |   |
||  |   |   |       LLUA   As String * 8     
||  |   |   |       |
||  |   |   |       |       Mode   As String * 8     
||  |   |   |       |       |
||  |   |   |       |       |       ReqMax As String * 5     
||  |   |   |       |       |       |
||  |   |   |       |       |       |       ResMax As String * 5 
||  |   |   |       |       |       |       |
||  |   |   |       |       |       |       |
VV  V   V   V       V       V       V       V

AB__CCCCDDDDEEEEEEEEFFFFFFFFGGGGGGGGHHHHH___IIIII___

因此 CharSet 应该是 Ansi,StructAlignment 应该是 4。使用 ByValTString 很好,使用 SizeConst 也是如此。

于 2012-06-19T12:57:43.643 回答
0

VB6 使用了 BSTR,不是吗?你不应该编组为 BSTR 吗?

于 2012-06-01T07:27:21.897 回答