射精一区欧美专区|国产精品66xx|亚洲视频一区导航|日韩欧美人妻精品中文|超碰婷婷xxnx|日韩无码综合激情|特级黄片一区二区|四虎日韩成人A√|久久精品内谢片|亚洲成a人无码电影

您現(xiàn)在的位置:首頁 > IT認(rèn)證 > 軟件水平 >

軟考程序員輔導(dǎo):c語言封送結(jié)構(gòu)體數(shù)組


2012年軟考程序員輔導(dǎo):c語言封送結(jié)構(gòu)體數(shù)組

在使用第三方的非托管API時,我們經(jīng)常會遇到參數(shù)為指針或指針的指針這種情況,

  一般我們會用IntPtr指向我們需要傳遞的參數(shù)地址;

  但是當(dāng)遇到這種一個導(dǎo)出函數(shù)時,我們?nèi)绾握_的使用IntPtr呢,

   extern “C” __declspec(dllexport) int GetClass(Class pClass[50]) ;

  由于這種情況也經(jīng)?赡苡龅剑晕抑谱髁2個示例程序來演示下如何處理這種非托管函數(shù)的調(diào)用!

  首先創(chuàng)建一個C++ 的DLL 設(shè)置一個如上的導(dǎo)出函數(shù)

   #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 的導(dǎo)出函數(shù)要求傳遞的參數(shù)為它自定義的Class結(jié)構(gòu)體數(shù)組, 那么我們在C#調(diào)用它時也要自定義對應(yīng)的結(jié)構(gòu)體了,

  我們可以定義為如下:

   [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個結(jié)構(gòu)體中的數(shù)組大小一定要跟C++中的限定一樣大小哦,接下來如何使用這個API來正確的獲取數(shù)據(jù)呢,大多數(shù)人可能想到像這樣的處理方式

   Class myclass = new Class();IntPtr ptr=Marshal.AllocHGlobal(Marshal.SizeOf(typeof(Class)));GetClass(ptr);Marshal.FreeHGlobal(ptr);沒錯,這樣的處理是沒問題的,但是我們的API的參數(shù)是Class數(shù)組,這種處理方式只是傳遞一個Class結(jié)構(gòu)體參數(shù),所以這種方式在這里就不太合適了,!

  那大家就想到先Class[] myclass = new Class[MaxClass]; 然后在用Marshal.AllocHGlobal 來獲取myclass 數(shù)據(jù)的指針,

  其實這樣也是錯的, 因為 Class結(jié)構(gòu)中包含了,不能直接封送的Student結(jié)構(gòu),所以無論如何上面的想法是錯誤的!

  那要怎么辦呢,其實很簡單,就是先分配一段非托管內(nèi)存,并調(diào)用API后,再將非托管內(nèi)容數(shù)據(jù)讀取到托管結(jié)構(gòu)體數(shù)據(jù)中!

  示例C語言封送結(jié)構(gòu)體數(shù)組演示代碼如下

   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語言封送結(jié)構(gòu)體數(shù)組,看看輸出結(jié)果是否正確!

相關(guān)文章

無相關(guān)信息
更新時間2022-03-13 11:10:58【至頂部↑】
聯(lián)系我們 | 郵件: | 客服熱線電話:4008816886(QQ同號) | 

付款方式留言簿投訴中心網(wǎng)站糾錯二維碼手機(jī)版

客服電話: