如何从C#转换(和马歇尔)CityHash std :: pair到C,反之亦然

CityHash允许我们生成128位哈希,但是将整数的128位表示形式定义为一对uint64(如header of CityHash.h所示):

typedef boost::uint64_t uint64;
typedef std::pair<uint64, uint64> uint128;

我有一个.NET包装器,它允许我调用64位版本的城市哈希:

public ref class CityHashDotNet
{
public:
    inline static void CityHash64Batch(const char* const value, uint64* const hashes, const int numValues)
    {
        // I have a wrapper for CityHash that allows me to make batch calls (saves on interops)
        CityHashWrapper::CityHash64Batch(value, hashes, numValues);
    }
    //...
}

这使我可以通过将值存储指针转换为sbyte *和哈希存储指针转换为ulong *并调用CityHashDotNet包装函数,轻松地从C#生成哈希:

// The MemoryPointer is an IntPtr
CityHashDotNet.CityHash64Batch((sbyte*)values.MemoryPointer, (ulong*)hashes.MemoryPointer, size);

我想对128位版本的city hash进行包装,但是我不知道如何对hash必需的std :: pair进行编组.我定义了一个与std :: pair相匹配的类,并且我打算使用它来复制std :: pair结构:

public class Pair<T1, T2>
{
    public T1 First { get; set; }
    public T2 Second { get; set; }

    public Pair(T1 first, T2 second)
    {
        First = first;
        Second = second;
    }
}

问题是我无法将IntPtr强制转换为std :: pair< ulong,ulong> *(ulong与uint64相同).我尝试将其强制转换为Pair< ulong,ulong> *,但出现构建错误:

// Does not work!
CityHashDotNet.CityHash128Batch((sbyte*)values.MemoryPointer, (Pair<ulong,ulong>*)hashes.MemoryPointer, size);

我得到的错误是:

error CS1503: ... cannot convert from `Pair<ulong,ulong>*` to `std.pair<unsigned __int64,unsigned __int64>*`

我如何解决这个问题?

解决方法:

您不能在指向这些类型的指针之间进行强制转换,因为它们的布局不兼容.您必须在每种类型的数组之间复制数据,一次复制一个字段.在C/C++LI代码中执行此操作,然后让C#代码看到漂亮的.NET数组.

上一篇:从C#传递到C代码的结构数组


下一篇:NET中有人使用Win32 API函数CredWrite吗?