c# Marshal 以 null 结尾的字符串数组
c# Marshal null-terminated string array
我正在尝试导入以下内容:
const char * const *object_get_prop_names(propobject_t *ocr);
如:
[DllImport("vender.dll", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Ansi)]
private static extern IntPtr object_get_prop_names(int* osr);
基于:https://limbioliong.wordpress.com/2011/08/14/returning-an-array-of-strings-from-c-to-c-part-1/
我试过以下方法:
var pNames = object_get_prop_names(hdl);
int StringCount = 200; //how do I know string count?
IntPtr[] pIntPtrArray = new IntPtr[StringCount];
ManagedStringArray = new string[StringCount];
Marshal.Copy(pNames, pIntPtrArray, 0, StringCount);
for (int i = 0; i < StringCount; i++)
{
ManagedStringArray[i] = Marshal.PtrToStringAnsi(pIntPtrArray[i]);
//Marshal.FreeCoTaskMem(pIntPtrArray[i]); crashes
}
//Marshal.FreeCoTaskMem(pUnmanagedStringArray); crashes
这行得通,但我猜我有内存泄漏并且正在访问我不应该访问的内存。
我应该如何释放内存?
我怎么知道计数?这是来自供应商,他们不会为小我修改 dll。 :)
也请原谅我钻研了我知之甚少的东西
遵循汉斯的建议:
var pNames = object_get_prop_names(hdl);
if (h == IntPtr.Zero)
{
return null;
}
var nameList = new List<string>();
int elementSize = Marshal.SizeOf(typeof(IntPtr));
for (int i = 0; i < 200; i++) //don't know length, pick large number
{
var ptr = Marshal.ReadIntPtr(pNames, i * elementSize);
var str = Marshal.PtrToStringAnsi(ptr);
if (!string.IsNullOrWhiteSpace(str))
{
nameList.Add(str);
}
else //end of pNames
{
break;
}
}
效果很好。我无法知道有多少 属性 个名字,所以我只需要选择一个比可能的 属性 个名字更大的数字。
我正在尝试导入以下内容:
const char * const *object_get_prop_names(propobject_t *ocr);
如:
[DllImport("vender.dll", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Ansi)]
private static extern IntPtr object_get_prop_names(int* osr);
基于:https://limbioliong.wordpress.com/2011/08/14/returning-an-array-of-strings-from-c-to-c-part-1/
我试过以下方法:
var pNames = object_get_prop_names(hdl);
int StringCount = 200; //how do I know string count?
IntPtr[] pIntPtrArray = new IntPtr[StringCount];
ManagedStringArray = new string[StringCount];
Marshal.Copy(pNames, pIntPtrArray, 0, StringCount);
for (int i = 0; i < StringCount; i++)
{
ManagedStringArray[i] = Marshal.PtrToStringAnsi(pIntPtrArray[i]);
//Marshal.FreeCoTaskMem(pIntPtrArray[i]); crashes
}
//Marshal.FreeCoTaskMem(pUnmanagedStringArray); crashes
这行得通,但我猜我有内存泄漏并且正在访问我不应该访问的内存。
我应该如何释放内存?
我怎么知道计数?这是来自供应商,他们不会为小我修改 dll。 :)
也请原谅我钻研了我知之甚少的东西
遵循汉斯的建议:
var pNames = object_get_prop_names(hdl);
if (h == IntPtr.Zero)
{
return null;
}
var nameList = new List<string>();
int elementSize = Marshal.SizeOf(typeof(IntPtr));
for (int i = 0; i < 200; i++) //don't know length, pick large number
{
var ptr = Marshal.ReadIntPtr(pNames, i * elementSize);
var str = Marshal.PtrToStringAnsi(ptr);
if (!string.IsNullOrWhiteSpace(str))
{
nameList.Add(str);
}
else //end of pNames
{
break;
}
}
效果很好。我无法知道有多少 属性 个名字,所以我只需要选择一个比可能的 属性 个名字更大的数字。