我环顾四周,似乎无法找到与我正在做的事情类似的解决方案。我有两个应用程序,一个 native C++ 应用程序和一个托管 C# 应用程序。 C++ 应用程序分配一个字节池,用于内存管理器。在此内存管理器中分配的每个对象都有一个 header ,每个 header 都有一个 char* 指向为该对象指定的名称。 C# 应用充当此内存的查看器。我使用内存映射文件来允许 C# 应用程序在 C++ 应用程序运行时读取内存。我的问题是我正在尝试从 header 结构中读取对象的名称并将其显示在 C# 中(或者只是将其存储在字符串中,无论如何)。使用不安全代码,我能够将构成 char*
的四个字节转换为 IntPtr
,将其转换为 void*
,然后调用 Marshal.PtrToStringAnsi
。这是代码:
IntPtr namePtr = new IntrPtr(BitConverter.ToInt32(bytes, index));
unsafe
{
void* ptr = namePtr.ToPointer();
char* cptr = (char*)ptr;
output = Marshal.PtrToStringAnsi((IntPtr)ptr);
}
在这种情况下,bytes
是从内存映射文件中读取的数组,表示 native 应用程序创建的所有字节池,index
是索引名称指针的第一个字节。
我已经验证,在事物的托管方面,调用 namePtr.ToPointer()
返回的地址正是 native 应用程序中名称指针的地址。如果这是 native 代码,我只需将 ptr
转换为 char*
就可以了,但是在我读过的托管代码中,我必须使用 Marshaller 来做这个。
这段代码会产生不同的结果。有时 cptr
为 null,有时它指向 \0
,有时它指向几个亚洲字符(当运行 PtrToStringAnsi
方法产生看似无关的字符)。我认为这可能是一个 fixed
的东西,但是 ToPointer
产生了一个固定的指针。有时在转换为 char*
之后,调试器会说 Unable to evaluate the expression。指针无效
或类似的东西(重现返回的每一个不同的东西并不容易)。有时我在读取内存时遇到访问冲突,这将我带到了 C++ 方面。
在 C++ 方面,我认为实际读取内存可能存在一些问题,因为尽管存储指针的内存是内存映射文件的一部分,但构成文本的实际字节不是。因此,我研究了如何更改对内存的读/写访问(请注意,在 Windows 上),并在 Windows 库中找到了 VirtualProtect 方法,我用它来将对内存的访问更改为 PAGE_EXECUTE_WRITECOPY,我认为它可以为任何应用程序提供有一个指向该地址的指针至少能够读取那里的内容。但这也没有解决问题。
简而言之:
我有一个指针(在 C# 中)指向 char 数组(在 C++ 应用程序中分配)中的第一个字符,我正在尝试将该 char 数组读入 C# 中的字符串。
编辑:
源头看起来像这样:
struct AllocatorHeader
{
// These bytes are reserved, and their purposes may change.
char _reserved[4];
// A pointer to a destructor mapping that is associated with this object.
DestructorMappingBase* _destructor;
// The size of the object this header is for.
unsigned int _size;
char* _name;
};
_name
字段是我试图在 C# 中取消引用的字段。
编辑:
截至目前,即使使用下面提供的解决方案,我也无法在托管代码中取消引用此 char*。因此,我只是在内存映射文件引用的池中复制了 char* 并使用了指向它的指针。这行得通,这让我相信这是一个与保护相关的问题。如果我在某个时候找到规避这种情况的方法,我将回答我自己的问题。在那之前,这将是我的解决方法。感谢所有提供帮助的人!
最佳答案
这在我的简单测试中似乎对我有用:
private static unsafe String MarshalUnsafeCStringToString(IntPtr ptr, Encoding encoding) {
void *rawPointer = ptr.ToPointer();
if (rawPointer == null) return "";
char* unsafeCString = (char*)rawPointer;
int lengthOfCString = 0;
while (unsafeCString[lengthOfCString] != '\0') {
lengthOfCString++;
}
// now that we have the length of the string, let's get its size in bytes
int lengthInBytes = encoding.GetByteCount (unsafeCString, lengthOfCString);
byte[] asByteArray = new byte[lengthInBytes];
fixed (byte *ptrByteArray = asByteArray) {
encoding.GetBytes(unsafeCString, lengthOfCString, ptrByteArray, lengthInBytes);
}
// now get the string
return encoding.GetString(asByteArray);
}
可能有点令人费解,但假设您的字符串以 NUL 结尾,它应该可以工作。
关于C# 字符* 到字符串,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19502012/