最近帮底层开发的同时用C#重新封装一下dll,也就是用C#类来封装C++Dll里的方法,以供用户使用。

之前也用到过类似的应用,大多数问题都出在类型转换上,但是这次的应用层出不穷,所以在这里总结一下,以供自己以后查阅,也希望对大家能够有所帮助。

  

首先,重复一下一些基本使用方法。具体的那些方式在这里就不重复讲了,网上很多的。比如http://blog.csdn.net/sunboyljp/archive/2009/12/31/5110639.aspx

c++ 头文件中的定义:

NPD_API int   NP_Init();

C#中定义函数

[DllImport("npd_api.dll")]

public static extern int NP_Init();

基本类型转换见下表(我用到过的):

BSTR——StringBuilder

LPCTSTR ——StringBuilder

LPCWSTR ——IntPtr

handle ——IntPtr

hwnd ——IntPtr

char *  ——string

int * ——ref int

int & ——ref int

void * ——IntPtrs

unsigned char * ——ref byte

BOOL ——bool

DWORD ——uint或int(我用的是uint,没出过什么问题)

我的问题来了,长期的经验教训我知道了:

1、指针做参数时在C#中一定要使用ref 或out关键字,尤其是结构体指针,要不会报内存读取错误,即使不报错数据也是不太对的。呵呵

   SIPCLIENT_API void WINAPI SCCleanup(SipClient * psip);

   [DllImport("sipclient.dll")]
       public static extern void SCCleanup(ref SipClient psip);

  其中SipClient是一个结构体。 

2、重写结构体的时候,之前有指明类型长度或数组长度的地方,也要进行相应的标注,要不也会导致内存错误。       

代码

 typedef struct {

     char sDVRIP[16]; /* DVR IP地址 */

     char sDVRIPMask[16]; /* DVR IP地址掩码 */

     DWORD dwNetInterface; /* 10M/100M自适应,索引 */

     WORD wDVRPort; /* 端口号 */

     BYTE byMACAddr[MACADDR_LEN]; /* 服务器的物理地址 */

 }NET_POSA_ETHERNET;

   public struct NET_POSA_ETHERNET
{
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
public string sDVRIP; //DVR IP地址
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
public string sDVRIPMask; // DVR IP地址掩码
public uint dwNetInterface; //网络接口 1-10MBase-T 2-10MBase-T全双工 3-100MBase-TX 4-100M全双工 5-10M/100M自适应
public uint wDVRPort; //端口号
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]
public byte[] byMACAddr; //[MACADDR_LEN]; //PPPoE用户名//服务器的物理地址
}

3、遇到这样一个问题,折腾了大半天时间——http://space.cnblogs.com/q/16616/。

  最后是在C++那边做了修改解决的,通过制定模块定义 (.def) 文件,统一制定导出函数对应的名称。返回值为结构体指针的函数用IntPtr也能使用了。  

代码

SIPCLIENT_API SipClient* SCInit(const char * reaml,   const char * from_ip, int from_port,   const char * to_ip, int to_port, const char * server_id,   const char * user_id, const char * user_name, void * user_obj_param);

[DllImport("sipclient.dll")]
public static extern IntPtr SCInit(string reaml, string from_ip, int from_port,
string to_ip, int to_port, string server_id,
string user_id, string user_name, IntPtr user_obj_param);

  IntPtr client = IntPtr.Zero;
client = SIPCLIENT_API.SCInit(REALM, CLIENT_IP, CLIENT_PORT, SERVER_IP,
     SERVER_PORT, SERVER_ID, USER_ID, USER_ID, IntPtr.Zero);
if (client != IntPtr.Zero)
sipclient = (SipClient)Marshal.PtrToStructure(client, typeof(SipClient));
else
MessageBox.Show("SipClient初始化失败!");

4、后来还遇到个回调函数导致的崩溃问题,又耽误了大半天时间,下班了还耽搁了会终于找的解决发办法了。

  刚开始同事分析出了崩溃的原因,都是回收方式惹的祸,可参见http://www.hudong.com/wiki/WINAPI,尝试使用__stdcall,但是还是没有解决问题

  后来实践证明,程序是很严谨的,半点差错都不能出才不会导致错误,思路还是__stdcall,只不过少改了东西,有两个地方需要改,才能保证不出错。

  参考http://hi.baidu.com/tease/blog/item/1fe7213802780f22b9998f5a.html。

  关键就是这两句话

  typedef void (_stdcall *CiCiCallBack) (bool started, void* client,char *message);
  将导出函数修改为:
  extern "C" _declspec(dllexport) bool _stdcall Test(char* fileName, CiCiCallBack callback)

  一开始的时候就只修改了定义那,却忘记了导出时的修改,差点就放弃了这条解决思路了,不过还好,所谓坚持就是胜利!

  
 5、后来封装好拿到用户那里用,却总是提示说找不到C++那些dll.

  网上一查,初步定位是开发环境引起的,跟环境部署有关系。我们的开发环境是vs2008,而客户使用的vs2010,通过几次尝试,问题终于了。

  首先考虑是缺少某些C++必备的运行库,存在相互依赖关系,所以导致找不到dll。用查看Dependency Walker查看才发现真的是客户机子上少了一些东西。

    但是此路不通,将缺少的那些东西拷贝到可执行程序目录下,问题依旧没有解决。但是依旧坚持这条路~

  尝试安装vcredist_x86.exe,以排除是否还是缺少了某些运行库的可能,问题依然存在。

  后来我想起来之前搜索问题的时候,看到好像跟dll的Releas\Debug版本还有关系,所有又尝试提议让同事将他们的c++dll改为Release版的。

    因为项目是多个人一起做了,编译Release版还花了不少时间,不过好歹问题终于解决了!

  总结:直接安装vcredist_x86.exe,所有dll必须使用Release版的。如果使用Debug版的就必须保证可执行程序目录下的dll是完整的,缺一不可!

  网上详细的讲解也很多,感觉这个总结的很好http://hi.baidu.com/fairysky/blog/item/e7a8366dbaa735f3431694c8.html。

做程序就怕出现问题,出现问题就怕不知道原因,知道原因了就好找解决的办法啦!

C#调用C++Dll封装时遇到的一系列问题相关推荐

  1. C#调用C++Dll封装时遇到的一系列问题 参考

     在合作开发时,C#时常需要调用C++DLL,当传递参数时时常遇到问题,尤其是传递和返回字符串是,现总结一下,分享给大家: VC++中主要字符串类型为:LPSTR,LPCSTR, LPCTSTR, ...

  2. [轉]c#简单调用DELPHI DLL封装窗体

    http://www.cnblogs.com/el-net/archive/2008/10/13/1309940.html dllimport Delphi dll dllimport dll Del ...

  3. c#调用c++dll接口及回调函数

    在合作开发时,C#时常需要调用C++DLL,当传递参数时时常遇到问题,尤其是传递和返回字符串是,现总结一下,分享给大家: VC++中主要字符串类型为:LPSTR,LPCSTR, LPCTSTR, st ...

  4. C#时常需要调用C++DLL

    在合作开发时,C#时常需要调用C++DLL,当传递参数时时常遇到问题,尤其是传递和返回字符串是,现总结一下,分享给大家: VC++中主要字符串类型为:LPSTR,LPCSTR, LPCTSTR, st ...

  5. C++ 中 对dll二次封装时,加载第三方库dll时 LoadLibrary 时错误代码126

    前提:  第三方dll名为 A.dll   我自己对A.dll又重新封装了一个B.dll  ,然后使用C程序进行测试,发现在B.dll中有加载A.dll的地方出错,错误代码为126. 具体代码如下: ...

  6. qt采用C++/CLI 方式调用C#dll的封装方法

    在qt中调用C++dll一般都可以直接使用,但是在调用C#版dll时,就有些麻烦了.本文采用C++/cli封装C#的dll的方式.实现了qt调用C#dll的方法. .h文件 #pragma once ...

  7. 多个类的DLL封装及调用

    网上大多是将函数封装成dll的教程,类的封装也是基本相似的. 创建DLL 在VS2010中新建一个win32->dll工程.如我建立的工程名为FaceDLL 添加facedll.h的头文件(里面 ...

  8. C/C++ DLL封装及调用

    目录 操作步骤 1. 创建dll 2. 创建测试程序 需求:C开发的控制台程序封装为dll,用于C++程序调用 开发环境:VS 2013 Pro 操作步骤 1. 创建dll 新建工程:C/C++ -& ...

  9. 调用matlab dll报错,c#调用MATLAB的dll时出错

    MATLAB是2014a,VS是2013的. 在C#中调用matlab生成的dll文件时,ceshi.Class1 yaoyi = new Class1();  对类进行实例时,抛出异常.异常为: 未 ...

最新文章

  1. pku 1185 炮兵阵地
  2. Nodejs 新特性 async await 的使用 以及使用 async await 处理异步
  3. 管程由哪三部分组成_黄蜀葵烘干机是由哪几个部分组成的
  4. 斗地主AI算法——第八章の被动出牌(2)
  5. 对人工智能产品发展的几点认识
  6. 暖冬福利放送,送3本技术好书
  7. 巨人java生鲜app下载_Java生鲜电商平台-生鲜电商平台SSO(单点登陆)技术方案(小程序/APP)...
  8. Java之小球碰撞反弹
  9. LoadRunner 录制IE 8卡死
  10. 单节点部署OpenStack平台及破解密码
  11. TeamViewer——一款强大的远程控制工具
  12. 迅雷离线下载与高速通道运作原理详解
  13. 序列化(boost serialization)
  14. 3. 自定义Java编译时注解处理器
  15. 求质数(素数)的方法
  16. 阿里云前端周刊 - 第 13 期 1
  17. 保险公司如何为数字化转型做准备
  18. 多级CIC滤波器的matlab仿真
  19. Centos 安装配置代理服务器软件Squid
  20. Python学习记录——Python容器:列表、元组、字典与集合(1)

热门文章

  1. Vue 教程第十七 篇—— Vuex 之 module
  2. Java线程:线程的同步与锁
  3. 施耐德电气:2016年数据中心的三大关注领域
  4. C#Redis集合set
  5. 《jQuery权威指南》学习笔记——第二章
  6. 关于linux特殊重定向的理解
  7. flaming pear flood中文版
  8. dreamweaver2020中文版
  9. go run go build go install 命令解释
  10. [转]为什么Java中的HashMap默认加载因子是0.75