【发布时间】:2012-01-26 03:29:01
【问题描述】:
我已经开始将 .NET 2.0 WinForms 应用程序升级到 .NET 4.0。好吧,升级过程只是切换平台目标的问题,但要让它真正工作。我想这就是它的全部了。
但 .NET 4.0 中的互操作性似乎发生了翻天覆地的变化。使用 DllImport(),应用程序嵌入了几个 Delphi dll。当应用程序以 .NET 2.0 为目标时,一切正常。但是当我将其更改为面向 .NET 4.0 时,事情开始变得混乱,就像某些东西正在破坏内存一样。
例如,它在奇怪的地方用“0”替换单个数字。在 IStream 中传递的数据将 8 个字符替换为 (Hex) 00 00 00 00 00 00 00 80,但只有大约 70% 的时间。两次连续调用检索相同的值会返回不同的结果(从内存中的缓存中检索值,第一次成功,第二次失败)。发送到日志的字符串显示被截断。
我尝试了很多方法来使调用约定更加明确,但都没有任何效果。所有字符串在 .NET 端都作为 [MarshalAs(UnmanagedType.LPWStr)] 处理,在 Delphi 端作为 PWChar 处理。
.NET 4.0 中的哪些变化会像这样破坏 P/Invoke?
--------------编辑------------------ -------------------
这是最简单的例子。它生成的 PDF 有时可以正常工作,但更常见的是损坏(并且在 .NET 2.0 中可以正常工作):
[DllImport(DLLName)]
public static extern void SetDBParameters(
[MarshalAs(UnmanagedType.LPWStr)] string Server,
[MarshalAs(UnmanagedType.LPWStr)] string Database,
[MarshalAs(UnmanagedType.LPWStr)] string User,
[MarshalAs(UnmanagedType.LPWStr)] string Password,
short IntegratedSecurity);
procedure SetDBParameters(Server, Database, User, Password: PWChar;
IntegratedSecurity: WordBool); stdcall;
[DllImport(DLLName)]
public static extern short GeneratePDF(
[MarshalAs(UnmanagedType.LPWStr)] string Param1,
[MarshalAs(UnmanagedType.LPWStr)] string Param2,
[MarshalAs(UnmanagedType.LPWStr)] string Param3,
[MarshalAs(UnmanagedType.LPWStr)] string Param4,
out IStream PDFData);
function GeneratePDF(Param1, Param2, Param3, Param4: PWChar;
out PDFData: IStream): WordBool; stdcall;
private byte[] ReadIStream(IStream Stream)
{
if (Stream == null)
return null;
System.Runtime.InteropServices.ComTypes.STATSTG streamstats;
Stream.Stat(out streamstats, 0);
Stream.Seek(0, 0, IntPtr.Zero);
if (streamstats.cbSize <= 0)
return null;
byte[] result = new byte[streamstats.cbSize];
Stream.Read(result, (int)streamstats.cbSize, IntPtr.Zero);
return result;
}
WordBool 和 short 最初是 boolean (Delphi) 和 bool (C#),我将它们更改为更明确,以防万一。
--------------编辑------------------ -------------------
我之前写的关于 WinForms 的内容似乎并不完全相关,我在没有任何 UI 的情况下重新创建了其中一个问题。下面程序在2.0/3.5下生成0,1,2,3,4,5,6,7,8,9,但是0,-1,-1,-1,-1,-1,-1,- 1,-1 低于 4.0。
using System;
using System.Runtime.InteropServices;
namespace TestNet4interop
{
static class Program
{
[DllImport("TestSimpleLibrary.dll", PreserveSig=true, CallingConvention = CallingConvention.StdCall)]
public static extern void AddToList(long value);
[DllImport("TestSimpleLibrary.dll", PreserveSig=true, CallingConvention = CallingConvention.StdCall)]
public static extern int GetFromList(long value);
static void Main()
{
for (long i = 0; i < 10; i++)
{
AddToList(i);
Console.WriteLine(GetFromList(i));
}
}
}
}
还有 Delphi 方面(用 Delphi 2007 编译):
library TestSimpleLibrary;
uses
SysUtils,
Classes;
{$R *.res}
var
List: TStringList;
procedure AddToList(value: int64); stdcall;
begin
List.Add(IntToStr(value));
end;
function GetFromList(value: int64): integer; stdcall;
begin
result := List.IndexOf(IntToStr(value));
end;
exports
AddToList,
GetFromList;
begin
List := TStringList.Create;
end.
【问题讨论】:
-
你能给我们看一下 PInvoke 签名吗?
-
这篇帖子 social.msdn.microsoft.com/Forums/en-US/csharpgeneral/thread/… 表明编组在 4.0 中确实发生了变化,但没有提供发生变化的完整列表。
-
也许很有趣:Visual Studio 2010 SP1 确实没有解决问题,我们只是在这里尝试过。