2015年软件水平考试程序员辅导笔记之C语言封送结构体数组知识
发布时间:2012/5/21 15:24:49 来源:城市网学院 编辑:ziteng
在使用第三方的非托管API时,我们经常会遇接事数为指针或指针的指针这种情形,
一般我们会用IntPtr指向我们需要传递的参数地址;
可是当碰着这种一个导出函数时,我们若何正确的使用IntPtr呢,
extern “C” __declspec(dllexport) int GetClass(Class pClass[50]) ;
因为这种情形也经常可能碰着,所以我建造了2个示例轨范来演示下若何措置这种非托管函数的挪用!
首先建树一个C++ 的DLL 设置一个如上的导出函数
#include #include typedef struct Student { char name[20];int age;double scores[32];}Student;typedef struct Class { int number;Student students[126];}Class;extern “C” __declspec(dllexport) int GetClass(Class pClass[50])
{ for(int i=0;i<50;i++)
{ pClass[i].number=i;for(int j=0;j<126;j++)
{ memset(pClass[i].students[j].name,0,20);sprintf(pClass[i].students[j].name,“name_%d_%d”,i,j);pClass[i].students[j].age=j%2==0?15:20;} return 0;}膳缦沔DLL 的导出函数要求传递的参数为它自界说的Class结构体数组, 那么我们在C#挪用它时也要自界说对应的结构体了,
我们可以界说为如下:
[StructLayout(LayoutKind.Sequential)] struct Student { [MarshalAs(UnmanagedType.ByValTStr,SizeConst=20)] public string name;public int age;[MarshalAs(UnmanagedType.ByValArray,SizeConst=32)] public double[] scores;} [StructLayout(LayoutKind.Sequential)] struct Class { public int number;[MarshalAs(UnmanagedType.ByValArray,SizeConst=126)] public Student[] students;}需要注重的是,这2个结构体中的数组巨细必然要跟C++中的限制一样巨细哦,接下来若何使用这个API来正确的获取数据呢,大大都人可能想到像这样的措置体例
Class myclass = new Class();IntPtr ptr=Marshal.AllocHGlobal(Marshal.SizeOf(typeof(Class)));GetClass(ptr);Marshal.FreeHGlobal(ptr);没错,这样的措置是没问题的,可是我们的API的参数是Class数组,这种措置体例只是传递一个Class结构体参数,所以这种体例在这里就不太合适了,!
那巨匠就想到先Class[] myclass = new Class[MaxClass]; 然后在用Marshal.AllocHGlobal 来获取myclass 数据的指针,
其拭魅这样也是错的, 因为 Class结构中包含了,不能直接封送的Student结构,所以无论若何膳缦沔的设法是错误的!
那要怎么办呢,其实很简单,就是先分配一段非托管内存,并挪用API后,再将非托管内容数据篡夺到托管结构体数据中!
示例C说话封送结构体数组演示代码如下
1 static void Main(string[] args)
2 {
3 int size = Marshal.SizeOf(typeof(Class)) * 50;
4 byte[] bytes = new byte[size];
5 IntPtr pBuff = Marshal.AllocHGlobal(size);
6 Class[] pClass = new Class[50];
7 GetClass(pBuff);
8 for (int i = 0; i < 50; i++)
9 {
10 IntPtr pPonitor = new IntPtr(pBuff.ToInt64() + Marshal.SizeOf(typeof(Class)) * i);
11 pClass[i] = (Class)Marshal.PtrToStructure(pPonitor, typeof(Class));
12 }
13 Marshal.FreeHGlobal(pBuff);
14 Console.ReadLine();
15 }有乐趣的不妨自己测试一下C说话封送结构体数组,看看输出结不美观是否正确!