C#和内存指针
关 键 词: serverword.net
虽然到了.net 的runtime 时代,C 仍然是不可磨灭的主要支柱,包括在MSDN 里见到的API 大部都是有指针类型存在,那么在以C 冠首的C# 里应当如何去使用指针呢?
其实,C# 里可以直接使用结构型的变量引用进API,但是毕竟它不是指针在对一些涉及以结构数组存放的内存区操作可能就力有不及了,我举一个例子要使用到结构数组的一个API(对这个API 不必加以了解,我们不是主说API):
net_api_status netshareenum(
LPWSTR servername,
DWORD level,
LPBYTE* bufptr,
DWORD prefmaxlen,
LPDWORD entriesread,
LPDWORD totalentries,
LPDWORD resume_handle
);
及它要用到的结构 (设 level 为1):
typedef struct _share_info_1 {
LPWSTR shi1_netname;
DWORD shi1_type;
LPWSTR shi1_remark;
} SHARE_INFO_1, *PSHARE_INFO_1, *LPSHARE_INFO_1;
在第三个参数 LPBYTE* bufptr 它就是一个指针,而非C 型传统语言的数组的做法,就可以直接声明成 ref SHARE_INFO_1[] bufptr,但我不推荐这么做,原因之一在C# 似乎还不支持结构数组块,而且它并没有存在数组型结构参数的重载。暂且不说如何去实现它,我希望以 C 的操作方式-指针去实现,所以我转成 C# 声明如下:
[structlayout(layoutkind.sequential)]
protected struct SHARE_INFO_1 {
[MarshalAs(UnmanagedType.LPWStr)] public string shi1_netname;
[MarshalAs(UnmanagedType.U4)] public uint shi1_type;
[MarshalAs(UnmanagedType.LPWStr)] public string shi1_remark;
}
[dllimport("netapi32.dll", entrypoint="netshareenum")]
protected static extern int NetShareEnum(
[MarshalAs(UnmanagedType.LPWStr)] string servername,
[MarshalAs(UnmanagedType.U4)] uint level,
ref IntPtr bufptr,
[MarshalAs(UnmanagedType.U4)] uint prefmaxlen,
[MarshalAs(UnmanagedType.U4)] ref uint entriesread,
[MarshalAs(UnmanagedType.U4)] ref uint totalentries,
[MarshalAs(UnmanagedType.U4)] ref uint resume_handle
);
在上面的函数执行完成后,会在指针 bufptr 所指的位置存在一个数据块,此内存块的数据格式是以 share_info_1 结构排列的(假设level为1),并且在 entriesread 中指示该块的结构个数,那么我们要如何将它转成真正可用的类型到 share_info_1[] shareinfo; 变量中去呢,其实 c# 提供了一种 marshal.ptrtostructure 的方法,它就是专门将指针所指的非托管内存块搬到托管内存中的指定的结构中去,可它不支持结构数组(目前.net 1.1 还没有该重载实现过),所以只好也最好一个一个的搬(而且一个一个搬更形象些),例子如下:
int32 ptr = bufptr.toint32();
for (int i=0; i<entriesread; i++) {
//开始将ptr 所指的内存块中搬一个SHARE_INFO_1 结构大小并按SHARE_INFO_1 格式排列的数据到 SHARE_INFO_1 型变量 shareInfo 中去
SHARE_INFO_1 shareInfo = (SHARE_INFO_1)Marshal.PtrToStructure(new IntPtr(ptr), typeof(SHARE_INFO_1));
ptr += Marshal.SizeOf(shareInfo); // 这是将指针向后移一个结构位,因为已经移完一个结构了
//todo: 操作当前的 shareinfo
}
上面的例子很明显了,bufptr 就是指针,该例子就是将内存中的指针所指的内存块(非托管内存)以一个指定结构的大小和排列格式移动.net 内存(托管内存)中的一个变量去,然后将指针向后移一个位置(ptr += marshal.sizeof(shareinfo)),此位置由指针当前位置加上 marshal.sizeof(shareinfo) 也就是已经移完的大小获得,它的意思其实就是将指针移到已经搬完的数据的后面去准备搬一下结构块
这个例子只是说明指针的用法,并不是主要说api 对该api 及它的参数不必细加研究