我正在调用一个非托管 C++ dll,它需要一个 char* 作为其参数之一,我想将一个 byte[] 推入其中。该项目是用 VB.NET 编写的。
什么类型的编组将为此工作?
我正在调用一个非托管 C++ dll,它需要一个 char* 作为其参数之一,我想将一个 byte[] 推入其中。该项目是用 VB.NET 编写的。
什么类型的编组将为此工作?
如果您需要固定托管结构以便将其作为参数传递,您可以使用以下代码。
// (c) 2007 Marc Clifton
/// <summary>
/// A helper class for pinning a managed structure so that it is suitable for
/// unmanaged calls. A pinned object will not be collected and will not be moved
/// by the GC until explicitly freed.
/// </summary>
internal class PinnedObject<T> : IDisposable where T : struct
{
protected T managedObject;
protected GCHandle handle;
protected IntPtr ptr;
protected bool disposed;
public T ManangedObject
{
get
{
return (T)handle.Target;
}
set
{
Marshal.StructureToPtr(value, ptr, false);
}
}
public IntPtr Pointer
{
get { return ptr; }
}
public int Size
{
get { return Marshal.SizeOf(managedObject); }
}
public PinnedObject()
{
managedObject = new T();
handle = GCHandle.Alloc(managedObject, GCHandleType.Pinned);
ptr = handle.AddrOfPinnedObject();
}
~PinnedObject()
{
Dispose();
}
public void Dispose()
{
if (!disposed)
{
if (handle.IsAllocated)
handle.Free();
ptr = IntPtr.Zero;
disposed = true;
}
}
}
}
然后,您可以使用 PinnedObject.Pointer 调用非托管代码。在您的 extern 声明中,使用 IntPtr 作为该参数的类型。
PinnedObject<BatteryQueryInformation> pinBatteryQueryInfo = new PinnedObject<BatteryQueryInformation>();
pinBatteryQueryInfo.ManangedObject = _structBatteryQueryInfo;
Unmanaged.Method(pinBatteryQueryInfo.Pointer);
在您的 PInvoke 定义中,只需将 char* 参数声明为 byte[] ,标准编组器将处理工作。
但这可能是也可能不是最好的主意。C++ 函数需要一个字符串还是需要一个数据缓冲区(C/C++ 代码经常使用 char* 作为缓冲区,依赖于 char 是一个字节的事实)?
如果它是一个缓冲区,那么 byte[] 肯定是正确的,但如果它需要一个字符串,那么如果将参数声明为字符串(要明确)并使用 Encoding.ASCII.GetString() 转换byte[] 到一个字符串。
此外,如果它的 C++ 函数需要一个字符串,并且您决定将参数声明为 byte[],请确保字节数组以零结尾,因为 C/C++ 就是这样确定字符串的结尾。
我不是 .net 专家,但我最近需要做一些类似的事情。
这不仅仅是序列化的问题,你还必须阻止垃圾收集器在 C++ 领域使用它时清理你的字节数组......
下面的 C# 片段应该会有所帮助。
// 固定 byte[] (byteArray) GCHandle 句柄 = GCHandle.Alloc(byteArray, GCHandleType.Pinned); IntPtr 地址 = handle.AddrOfPinnedObject(); // 使用地址指针做你的 C++ 工作。 // 清理 处理.Free();