gpt4 book ai didi

c# - 从 C# 调用带有 C++ header 的 dll

转载 作者:塔克拉玛干 更新时间:2023-11-03 02:24:27 25 4
gpt4 key购买 nike

我正在尝试在我有 C++ header 的 dll 中调用一个方法。我正在从 C# 调用 dll。输入是字符串,输出是二进制数据。以下 3 种方法中的任何一种都可能起作用,我只是不知道如何让它们中的任何一种一直起作用。C# 声明是我做的,所以它们可能不正确

1:我可以获取到hGlobal,但是我不知道如何从句柄中获取数据。

//CMBT_LL_WINAPI INT DLLPROC  LlConvertStringToHGLOBALW(LPCWSTR pszText, _PHGLOBAL phMemory);
[DllImport("cmll15.dll", EntryPoint = "LlConvertStringToHGLOBALW", CharSet = CharSet.Unicode, ExactSpelling = true)]
private static extern int _LlConvertStringToHGlobal32(string text, ref IntPtr handle);

2:

[DllImport("cmll15.dll", EntryPoint = "LlConvertStringToBLOBW", CharSet = CharSet.Unicode, ExactSpelling = true)]
//CMBT_LL_WINAPI INT DLLPROC LlConvertStringToBLOBW(LPCWSTR pszText, _PUINT8 pBytes, UINT nBytes);
private static extern int _LlConvertStringToBLOBW(string text, ref IntPtr pBytes, UInt32 nBytes);

3:

[DllImport("cmll15.dll", EntryPoint = "LlConvertStringToStreamW", CharSet = CharSet.Unicode, ExactSpelling = true)]
//CMBT_LL_WINAPI INT DLLPROC LlConvertStringToStreamW(LPCWSTR pszText, _PISTREAM pStream);
private static extern int _LlConvertStringToStreamW(string text, ref IntPtr pStream);

已更新,这是我认为最终会得到的代码。

    [DllImport("kernel32.dll", SetLastError = true, ExactSpelling = true)]
private static extern UIntPtr GlobalSize(IntPtr hMem);

[DllImport("kernel32.dll", SetLastError = true, ExactSpelling = true)]
public static extern IntPtr GlobalLock(IntPtr handle);

[DllImport("kernel32.dll", SetLastError = true, ExactSpelling = true)]
public static extern IntPtr GlobalUnlock(IntPtr handle);

[DllImport("cmll15.dll", EntryPoint = "LlConvertStringToHGLOBALW", CharSet = CharSet.Unicode, ExactSpelling = true)]
private static extern int _LlConvertStringToHGlobal32(string text, ref IntPtr handle);

private static void Main(string[] args)
{
IntPtr dataHandle = IntPtr.Zero;
_LlConvertStringToHGlobal32(Contents, ref dataHandle);
try
{
var size = (uint) GlobalSize(dataHandle);
var array = new byte[size];
IntPtr dataPtr = GlobalLock(dataHandle);
try
{
Marshal.Copy(dataPtr, array, 0, array.Length);
}
finally
{
GlobalUnlock(dataPtr);
}

using (var fs = new FileStream("c:\\file.dat", FileMode.Create))
{
fs.Write(array, 0, array.Length);
}
}
finally
{
Marshal.FreeHGlobal(dataHandle);
}
}

最佳答案

第一个应该是最容易开始的,因为它留给了被调用者来计算所需的大小。然而,您应该如何知道分配的大小并不明显。也许返回值。您始终可以调用 GlobalSize() 以从 HGLOBAL 句柄获取大小。您应该调用 GlobalLock() 将句柄转换为指针,然后调用 Marshal.CopyMemory() 将其复制到 byte[]。通过调用 GlobalUnlock() 和 Marshal.FreeHGlobal() 来释放内存进行清理,将其放在 finally block 中,这样就不会泄漏。

对于第二个,您应该将第二个参数声明为 byte[](而不是 ref)。问题是您必须预先猜测数组的大小。失败模式是猜测尺寸太小。

第三个需要 COM IStream。将其声明为 System.Runtime.InteropServices.ComTypes.IStream。它的行为很像 .NET Stream,您可以调用 Seek 来查找开头并调用 Read 来读取数据。

我会选择第一个,最不可能在你脸上爆炸。

关于c# - 从 C# 调用带有 C++ header 的 dll,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4666849/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com