欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

C语言封送结构体数组

程序员文章站 2023-01-13 09:09:08
在使用第三方的非托管api时,我们经常会遇到参数为指针或指针的指针这种情况,   一般我们会用intptr指向我们需要传递的参数地址;   但是当遇到这种一个导出函数时,我们如...

在使用第三方的非托管api时,我们经常会遇到参数为指针或指针的指针这种情况,
  一般我们会用intptr指向我们需要传递的参数地址;

  但是当遇到这种一个导出函数时,我们如何正确的使用intptr呢,

  extern "c" __declspec(dllexport) int getclass(class pclass[50]) ;

  由于这种情况也经常可能遇到,所以我制作了2个示例程序来演示下如何处理这种非托管函数的调用!

  首先创建一个c++ 的dll  设置一个如上的导出函数

  #include <windows.h> #include <stdio.h> 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语言封送结构体数组,看看输出结果是否正确!

 

摘自 脉凌网络