zoukankan      html  css  js  c++  java
  • C#调用C++动态库方法及动态库封装总结

      我只是粗浅的学习过一些C++语法, 变量类型等基础内容, 如有不对的地方还望指出. 如果你跟我一样, 对指针操作不了解, 对封装C++动态库头疼的话, 下面内容还是有帮助的.
       转载请注明出处: http://www.cnblogs.com/zaiyuzhong/p/Csharp-package-Cplusplus-dll.html

      首先给一个类型转换的表, 这个表可能跟使用时具体情况有关, 仅供参考, 点击这里查看.

      1. C++变量类型大小写问题:
      一种是小写的, 比如char. 这种是系统定义的(关键字); 另一种是CHAR, 这个是自定义的, 有说这个是宏, 有说这个在VC++里是.Net定义的, 具体怎样我还没遇到过, 没有例子供研究. C++是大小写敏感的.

      2. int的长度问题:
      int的长度和编译器是有关系的. 如果动态库和封装代码由同一个编译器编译运行时, 可以不考虑转换int长度.
      
      3. 结构体成员的长度问题:
    不好描述, 举个例子:
      C++代码:
      typedef struct Result
      {
        char color[8];
      }
      对应C#代码:
      public struct Result
      {
        [MarshalAs(UnmanagedType.ByValArray, SizeConst = 8)]
        public byte[] color;
      }
      //C#中public修饰符可根据需要修改为其他访问修饰符.
    在调用前可先对数据长度进行验证后再调用.
      
      4. 指针问题:
      有两种情况, 一种是封装C++函数时, 参数有指针类型的, 如果该参数是引用类型, 那么在C#中使用该参数时传递的也是变量指针(引用), 所以不用处理; 如果该参数是结构类型, 如int* , 封装时需要表示为ref, 例如:
    C++: int InitSDK(Config* pConfig); //Config是结构体;
    C# : public int InitSDK(ref Config pConfig); //int类型是否修改参见上述问题2;

      另外一种情况比较麻烦, C++函数需要给他分配一块内存空间:
    C++: int RecogImage(const unsigned char* pbyBits, Result* pResult);
       // pbyBits[in] 指向内存图像数据的指针,数据格式为输入图像的格式;
       // pResult[out] 识别结果数组, 调用方开辟pResult[nResultNum]内存;(注: 这两行注释是文档上的函数说明)
    Result是动态库定义的结构体, nResultNum是自己定义的变量, 数值与这里无关, 测试后发现声明参数类型为数组是不行的.
    C#: [DllImport("dll路径")]
      public static extern int RecogImage(IntPtr pbyBits, IntPtr pResult);
      // 这里的IntPtr虽然也是结构体, 但由于它本身就代表一个指针所以不用ref;

    这个方法的调用方法如下, 开辟内存和读取识别结果麻烦点:

     1 using System.Drawing;
     2 using System.Drawing.Imaging;
     3 
     4 //开辟识别结果数组内存
     5 int nResultNum = 6; // 数值与代码无关
     6 var result = new Result();
     7 var rSize = Marshal.SizeOf(result);
     8 var resultsPtr = Marshal.AllocHGlobal(rSize * nResultNum);
     9 
    10 //获取图像数据指针
    11 var image = new Bitmap("路径");
    12 var bmpData = image.LockBits(new Rectangle(0, 0, image.Width, image.Height), ImageLockMode.ReadOnly, PixelFormat.Format24bppRgb);
    13 
    14 //识别图像
    15 var r = PlateSDK.RecogImage(bmpData.Scan0, resultsPtr);
    16 
    17 //读取识别结果
    18 if(r != 0) return; // 识别失败, r为错误编号;
    19 else
    20 {
    21   var results = new Result[nResultNum];
    22   for(int i = 0; i < nResultNum; i++)
    23     results[i] = (Result)Marshal.PtrToStructure(results + rSize * i, typeof(Result));
    24 } // results 就是识别结果数组
    

     嗯, 其中3/4是我遇到比较麻烦的, 需要熟悉 Marshal(msdn), IntPtr(msdn) 和 MarshalAs(msdn).

  • 相关阅读:
    Why use strong named assemblies?
    Dependency Walker
    “等一下,我碰!”——常见的2D碰撞检测
    MOBA游戏的网络同步技术
    VS2017如何配置openGL环境
    UE4关于Oculus Rift (VR)开发忠告
    UE4 的json读写方式
    Blueprint 编译概述
    UE4编码规范
    Unreal Enginer4特性介绍
  • 原文地址:https://www.cnblogs.com/zaiyuzhong/p/Csharp-package-Cplusplus-dll.html
Copyright © 2011-2022 走看看