2

我正在调用一个非托管 C++ dll,它需要一个 char* 作为其参数之一,我想将一个 byte[] 推入其中。该项目是用 VB.NET 编写的。

什么类型的编组将为此工作?

4

3 回答 3

1

如果您需要固定托管结构以便将其作为参数传递,您可以使用以下代码。

    // (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);
于 2008-11-05T08:25:56.753 回答
0

在您的 PInvoke 定义中,只需将 char* 参数声明为 byte[] ,标准编组器将处理工作。

但这可能是也可能不是最好的主意。C++ 函数需要一个字符串还是需要一个数据缓冲区(C/C++ 代码经常使用 char* 作为缓冲区,依赖于 char 是一个字节的事实)?

如果它是一个缓冲区,那么 byte[] 肯定是正确的,但如果它需要一个字符串,那么如果将参数声明为字符串(要明确)并使用 Encoding.ASCII.GetString() 转换byte[] 到一个字符串。

此外,如果它的 C++ 函数需要一个字符串,并且您决定将参数声明为 byte[],请确保字节数组以零结尾,因为 C/C++ 就是这样确定字符串的结尾。

于 2009-11-21T13:52:49.437 回答
0

我不是 .net 专家,但我最近需要做一些类似的事情。

这不仅仅是序列化的问题,你还必须阻止垃圾收集器在 C++ 领域使用它时清理你的字节数组......

下面的 C# 片段应该会有所帮助。

// 固定 byte[] (byteArray)  
GCHandle 句柄 = GCHandle.Alloc(byteArray, GCHandleType.Pinned);   
IntPtr 地址 = handle.AddrOfPinnedObject();  
// 使用地址指针做你的 C++ 工作。

// 清理
处理.Free();
于 2008-11-05T04:45:54.593 回答