C#调用C++写的DLL总结

前几天做了一个 C++ 的 DLL,供网页调用,网页是用 C# 做的。
C++ 的 DLL 做起来简单,同时完成了一个 C++ EXE 调用 DLL 进行了调试。一切 OK!
然后将 DLL 转到做 C# 处进行测试,发现要不调用失败,要不得不到数据。
C# 调用 C++ 的 DLL 真的这样麻烦?

C++ 的 DLL 提供一个功能,将一字符串经过转换后形成另一字符串,然后在网页上显示转移后的此字符串。
C++ 的接口开始时设计为:

1 /*
2  * 功能:
3  * 参数: (in) pcInStr 用户输入的字符串
4  * (in) iInLen 用户输入的字符串的长度
5  * (out) pcOutStr 输出的字符串
6  * (in)  iLen 输出的字符串的最大长度
7 */
8 BOOL Func(unsigned char *pcInStr,int iInLen,unsigned char *pcOutStr,int iLen);

此函数只有一个返回值,通过第三个参数:pcOutStr,其它都是输入的。

C# 按此接口,调用方法如下:

1 public static extern char[] Func(ref char[] pcInStr, ref int iInLen,ref char[] pcOutStr,ref int iLen);

编译没有问题,但运行时发现,通过 iInLen 和 iLen 传入的值不对。
因为 DLL 中的函数进行了参数有效性判断,不是有效的长度会直接返回。
为测试增加了 MessageBox 显示此两个 Len 的值,提示此两个参数的值好像是随机数一样,每次运行的数值都不相同,但传入的数据始终都是固定的。
如果除第三个参外不使用 ref,传入的 Len 数值是没有问题的。但运行时会产生异常!
查了半天未找出原因,无奈之下,只好修改 DLL 的接口:不用参数来传值,修改为使用返回值。
DLL 中的 C++ 接口修改为:

1 char *Func(unsigned char *pcInStr,int iInLen);

C# 按此接口,调用方法试了以下两种:

1 (1) public static extern char[] Func(char[] pcInStr, int iInLen);
2 (2) public static extern string Func(char[] pcInStr, int iInLen);

都在运行时执行到调用 DLL 的接口时,发生异常。
写了个 C# 的 From 调用 DLL 的测试程序,同样发生异常,异常的信息为:
Additional information: Cannot marshal ‘return value‘: Invalid managed/unmanaged type combination.

最后,将调用方法修改为:

1 public static extern StringBuilder Func(char[] pcInStr, int iInLen);

这样调用,调用 DLL 的接口时,不再发生异常,但返回值一直为空。
怀疑是 DLL 中的接口未返回值,因此在 DLL 中的接口最后返回数据前,增加一 MessageBox 来显示返回的数据。
运行后发现,DLL 中的 C++ 接口通过 C# 调用后,MessageBox 显示了正常的数据。与 C++ EXE 调用 DLL 时显示的数据一致。
但 C++ EXE 可以得到数据,C# 的网页程序却只能得到一个 NULL 值。

最后怀疑是 C++ 和 C# 的程序运行的空间不同,导致此问题。
因为在 DLL 的 C++ 接口中,返回值定义在接口函数中,是一个数组;怀疑数组在 C# 调用返回时,已经被释放。
将此数组定义从接口中移到接口外,定义成一全局变量,其它未做改变。
此时,发现 C# 的网页程序调用 C++ DLL 中的接口后,也可以得到正确的数据。

总结:
C# 调用 C++ DLL 时,需要注意两个问题:
(1) 数据类型之间的转换,特别是指针;
(2) 运行作用域。
有一个问题一直未解决,就是 ref 传值的问题。为什么通过 ref 传 int 型的值到 C++ DLL 中,数据会发生变化?
偶基本上未用过 C# ,也无法解决。

最后的解决办法:恢复最初四个参数的接口,C# 的声明接口按如下定义:

1 public static extern bool Func(char[] pcInStr, int iInLen, [Out,MarshalAs(UnmanagedType.LPArray)]char[] pcOutStr, int iLen);
时间: 2024-08-08 09:33:21

C#调用C++写的DLL总结的相关文章

C#程序调用C++写的dll传递string出现bad ptr

本来是做C/C++的,因为项目需要,所以才搞的C#,说实话,很鄙视做C#的,总结起来,扯淡的DllImport,有本事别用这破玩意,看你C#还能干啥? 参考网上的按照下面的方式来,结果在C++的dll库中打断点,第二个参数怎么都是bad ptr,郁闷至极,耗费了两天的功夫,都没有搞定,也参考了:http://blog.csdn.net/yongshengsilingsa/article/details/7917877 的文章,也无济于事.最后就索性自己再写个简单的dll和exe,只有一个接口,调

Delphi7程序调用C#写的DLL解决办法(转)

近来,因工作需要,必须解决Delphi7写的主程序调用C#写的dll的问题.在网上一番搜索,又经过种种试验,最终证明有以下两种方法可行:    编写C#dll的方法都一样,首先在vs2005中创建一个"类库"项目TestDll,using System.Runtime.InteropServices;namespace TestDll{public interface  I TestClass  {     void YourProcedure(stirng param1);}[Cla

c++ c# java 调用 c++ 写的dll

1. vs 中新建win32 dll 项目   testdll 添加实现文件       test.cpp #include "stdafx.h" #include <iostream>using namespace std;int Add(int plus1, int plus2){ int add_result = plus1 + plus2; return add_result;} 添加模板定义文件 LIBRARY "testdll"EXPORTS

发现个delphi调用vc写的Dll中包含pchar参数报错奇怪现象

发现个delphi调用vc写的Dll中包含pchar参数奇怪现象 procedure中的第一行语句不能直接调用DLL的函数,否则会运行错,在之前任意加上条语句就不报错了奇怪! vc的DLL源码地址 http://blog.csdn.net/lqena/article/details/46357165 Delphi源码如下: unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Co

lua调用C++写的DLL实现“热更新”

原创作品,转载请注明来源是CSDN:http://blog.csdn.net/relar/article/details/38084689 开发游戏服务器往往有"热更新"的需求,就是在不停止服务程序的情况下,对服务程序进行升级.这里采用lua脚本桥接C++的模式.程序主框架用C++,程序的业务逻辑也是C++(具体的是C++写的DLL),这两者之间用LUA脚本语言进行桥接.当程序运行时,只要改变LUA脚本,即可以选择使用不同的DLL,以实现业务逻辑的升级更新. 上演示代码,代码分为三部分

QT调用C#写的Dll

参见: https://blog.csdn.net/weixin_42420155/article/details/81060945 C#写的dll是没有dllMain入口函数的,是一种中间语言,需要.Net运行时进行做本地化工作,因此如果要调用C#写的dll,需要依赖.Net运行时,然而Qt中还无法直接调用.Net运行时,最好的方式是能够在Qt中直接调用C#dll的函数,但是Qt明显只能调用C++写的dll,所以就只能通过编写一个C++的dll导出接口供Qt调用,这个C++编写的dll对C#写

Unity中C#调用C++写的DLL之Swig篇

hash索引btree索引聚簇索引非聚簇索引 安装AndroidSDK的一些坑与注意点 近来要用Unity打包到安卓上玩, Unity那边需要用到服务器中用C++写的库,对比了 P/Invoke 和 C++/CLI 两种方式, 都不够省心省力, 决定使用 Swig来撸. 教程基本上按照这篇文章就可以, 文章写得非常详尽, 但文中关于设置 swiglib.i 自定义生成工具的命令行的时候, 他文中的下面一段要注意 : 在常规中选择命令行并且写入: echo on $(SolutionDir)/..

Delphi调用C++写的dll示例

最近做一个读市民卡的项目,读卡器公司提供的读市民卡dll是用C++写的. 下面记录一些自己的心得,供需要的朋友参考. 声明dll函数要加上stdcall关键字,否则可能会报地址非法的错误. 代码: unit cMain; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, DB, ADODB, StdCtrls, ComCtrls,ActiveX, E

通过COM组件方式实现java调用C#写的DLL文件

通过Java调用C#的DLL文件实现.网上资料很多,自己整合总结了一下,做个备忘. 一.C#写com组件 开发环境 VS2013 1.新建工程:TestDemo(类库项目)     //根据自己需要取工程名 注:WIN7以上系统VS必须以管理员身份启动!!!!! 2.右键点击工程->应用程序->程序集信息->使程序集com可见,打上勾(或者:项目->TestDemo属性->应用程序->程序集信息->使程序集com可见,打上勾) 注:1.创建工程的时候,NET Fr