C#委托实现C++ Dll中的回调函数

from:https://blog.csdn.net/ferrycooper/article/details/63261771

很多的Dll都是CC++写的,那么如果C#想要调用Dll中的函数怎么办,尤其是Dll函数其中一个参数是函数指针的,即里面有回掉函数的用C#怎么实现?

C中的回掉函数在C#中有中特殊的处理方式叫委托,即要实现的回掉函数委托给另一个和它返回值类型以及函数参数类型、数量一样的方法来实现。


一、新建项目Visual C++  Win32控制台应用,工程名为CcreateDll,解决方案名为Dlltest

 

确定—>下一步

 

 

应用程序类型选Dll>完成

 

新建头文件Ccreate.h,声明导出函数,其中API_DECLSPEC int CallPFun(addP callback, inta, int b) 第一个参数为函数指针,内容如下:

 

[cpp] view plain copy
  1. #pragma once  
  2.   
  3. #ifndef Ccreate_H_  
  4. #define Ccreatel_H_  
  5.   
  6. typedef  int(*addP)(intint);  
  7.   
  8. #ifdef _EXPORTING   
  9. #define API_DECLSPEC extern "C" _declspec(dllexport)   
  10. #else   
  11. #define API_DECLSPEC  extern "C" _declspec(dllimport)   
  12. #endif  
  13.   
  14. API_DECLSPEC int Add(int plus1, int plus2);  
  15. API_DECLSPEC int mulp(int plus1, int plus2);  
  16. API_DECLSPEC int CallPFun(addP callback, int a, int b);  
  17.   
  18. #endif  

头文件有了,在CcreateDll.cppinclude头文件,并实现相关函数。Ccreate.cpp如下

 

[cpp] view plain copy
  1. // CcreateDll.cpp : 定义 DLL 应用程序的导出函数。  
  2. //  
  3.   
  4. #include "stdafx.h"  
  5. #include <iostream>   
  6. #include "Ccreate.h"  
  7.   
  8. using namespace std;  
  9.   
  10. int Add(int plus1, int plus2)  
  11. {  
  12.     int add_result = plus1 + plus2;  
  13.     return add_result;  
  14. }  
  15. int mulp(int plus1, int plus2)  
  16. {  
  17.     int add_result = plus1 * plus2;  
  18.     return add_result;  
  19. }  
  20.   
  21. int CallPFun(int(*callback)(intint), int a, int b) {  
  22.     return callback(a, b);  
  23. }  

函数CallPFun实际就是传入函数指针及其参数,内部直接调用函数指针。

Release模式下生成CcreateDll工程

 

生成成功后在解决方案目录的Release文件夹下会看到生成的CcreateDll.dll,使用Dll查看工具可以看到三个导出函数。

 

二、新建C#控制台工程CsharpCallDll实现调用Dll并使用委托实现回掉。

 

 

CsharpCallDll工程Program.cs如下:

 

[csharp] view plain copy
  1. using System;  
  2. using System.Collections.Generic;  
  3. using System.Linq;  
  4. using System.Text;  
  5. using System.Threading.Tasks;  
  6. using System.Runtime.InteropServices;  
  7.   
  8. namespace CsharpCallDll  
  9. {  
  10.     public class Program  
  11.     {  
  12.         [UnmanagedFunctionPointer(CallingConvention.Cdecl)]  
  13.         public delegate int DllcallBack(int num1, int num2);  
  14.   
  15.         [DllImport(@"../../../Release/CcreateDll.dll", EntryPoint = "Add", SetLastError = true, CharSet = CharSet.Ansi, ExactSpelling = false, CallingConvention = CallingConvention.Cdecl)]  
  16.         extern static int Add(int a, int b);  
  17.   
  18.         [DllImport(@"../../../Release/CcreateDll.dll", EntryPoint = "mulp", SetLastError = true, CharSet = CharSet.Ansi, ExactSpelling = false, CallingConvention = CallingConvention.Cdecl)]  
  19.         extern static int mulp(int a, int b);  
  20.   
  21.         [DllImport(@"../../../Release/CcreateDll.dll", EntryPoint = "CallPFun", SetLastError = true, CharSet = CharSet.Ansi, ExactSpelling = false, CallingConvention = CallingConvention.Cdecl)]  
  22.         public extern static int CallPFun(DllcallBack pfun, int a, int b);  
  23.         //[MarshalAs(UnmanagedType.FunctionPtr)]  
  24.         static void Main(string[] args)  
  25.         {  
  26.             int a = 3;  
  27.             int b = 4;  
  28.             int result;  
  29.             DllcallBack mycall;  
  30.             mycall = new DllcallBack(Program.CsharpCall);  
  31.             result = Add(a, b);  
  32.             Console.WriteLine("Add 返回{0}", result);  
  33.             result = mulp(a, b);  
  34.             Console.WriteLine("mulp 返回{0}", result);  
  35.             result = CallPFun(mycall, a, b);  
  36.             Console.WriteLine("dll回掉 返回{0}", result);  
  37.             Console.ReadLine();  
  38.         }  
  39.   
  40.         public static int CsharpCall(int a, int b)  
  41.         {  
  42.             return a * a + b * b;  
  43.         }  
  44.     }  
  45. }  

    

通过DllImport导入相应的Dll并声明Dll中的导出函数,CcreateDll.dll中导出函数CallPFun有三个参数,原型为

 

[cpp] view plain copy
  1. int CallPFun(int(*callback)(intint), int a, int b) {  
  2.     return callback(a, b);  
  3.    }  
   

参数1为一个带两个int参数的返回值为int型的函数指针,这里声明一个委托

public delegate int DllcallBack(int num1, intnum2);

该委托可以指向任何带两个int型参数且返回值为int型的方法,这里的CsharpCall方法可以看作是回掉函数的实现。

 

[csharp] view plain copy
  1. public static int CsharpCall(int a, int b)  
  2.    {  
  3.             return a * a + b * b;  
  4.    }  
   

通过        DllcallBack mycall;

           mycall = new DllcallBack(Program.CsharpCall);

  把实际要完成的工作交给CsharpCall去完成。

    运行CsharpCallDll,结果如下:

  

 

是不是实现了C#委托实现回掉

 

最后还有如果声明委托时在public delegate int DllcallBack(int num1, int num2);上面没有[UnmanagedFunctionPointer(CallingConvention.Cdecl)]这一句,那么运行时将会出现System.AccessViolationException异常,如下

 

 

还有Dll调用约定,CallingConvention.有五种调用方式

CallingConvention= CallingConvention.StdCall

CallingConvention= CallingConvention.Cdecl

CallingConvention= CallingConvention.FastCall

CallingConvention= CallingConvention.ThisCall

CallingConvention= CallingConvention.Winapi

到底使用哪种方式,网上有说"Bydefault, C and C++ use cdecl - but marshalling uses stdcall to match theWindows API."即默认情况下,CC++使用的Cdecl调用,但编组使用StdCall调用匹配的Windows API,对于FastCallThisCallWinapi这三种调用方式尚不清楚。

这里将CallingConvention= CallingConvention.Cdecl改成CallingConvention = CallingConvention.StdCall,重新运行导致堆栈不对称如下

 



本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/458434.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

安装Birt方法

安装BIRT 方法&#xff1a; 博客地址&#xff1a;http://www.mamicode.com/info-detail-850588.html 注意&#xff1a;在 Install new Software 中输入地址&#xff1a;http://download.eclipse.org/birt/update-site/4.2-interim 看好了 出来的四项要全部选中 &#xff0c;然后…

iOS NSString和NSDate转换

后台返回的时间字符串不是标准的时间而是计算机时间的时候&#xff0c;我们需要将它们转换为标准时间&#xff0c;再进行转换。 //字符串转为时间&#xff0c;时间格式自己定 NSString * time "1501776000"; //时间字符串 NSInteger num [time integerValue]; …

15个最好的HTML5前端响应式框架(2014)

文中的多个框架基于SASS创建&#xff0c;SCSS是一种比LESS更简洁的样式表编程语言&#xff0c;它能够编绎成CSS&#xff0c;可复用CSS代码&#xff0c;声明变量&#xff0c;甚至是函数&#xff0c;类Ruby/Python的语法。參见&#xff1a; LESS vs SASS&#xff1f;选择哪种CSS样…

DLL导出类和导出函数

from:https://blog.csdn.net/goodluckmt/article/details/526912971、动态库DLL中的类或者函数有时候要被其他的库调用&#xff0c;因此需要被其他库调用的类或者函数需要进行导出。 2、首先编写需要导出的DLL&#xff0c;新建一个工程设置应用程序类型为DLL3、类的代码如下 头…

Nginx做web服务器linux内核参数优化

Nginx做web服务器linux内核参数优化Nginx提供web服务时Linux内核参数调整是必不可少的&#xff0c;其中在优化方面就需要我们格外的注意。在下面就是对Linux内核参数优化的详细介绍&#xff0c;希望大家有所收获。关于Linux内核参数的优化&#xff1a;net.ipv4.tcp_max_tw_buck…

iOS系统发布时间

发布日期 版本编号 更改2017年9月12日 iOS11 GM seed2017年6月19日 iOS10.3.32016年11月9日 iOS10.2 Beta2 发布iOS10.2测试版版2016年11月1日 iOS10.2 Beta1 发布iOS10.2测试版版2016年10月25日 iOS10.1 发布iOS10.1正式版2016年9月22日 iOS10.1 Beta 发布iOS10.1测试版2016年…

Tomacat服务器的安装和配置

一&#xff0c; Tomcat服务器的下载地址(Apache Tomcat的官网): http://tomcat.apache.org/download-70.cgi 这里为了稳定性安装的版本为7.0. 截止目前版本已经发行到了9.0. 二&#xff0c; 解压下载的压缩包之后直接进入傻瓜式的安装。   下载文件的全名&#xff1a; apach…

函数指针与回调函数详解

函数指针与回调函数详解 1.什么是函数指针&#xff1f; 函数&#xff08;的&#xff09;指针就是指针。这个指针存放一个函数的地址&#xff0c;而函数的名称就该函数的入口&#xff0c;即地址。这类似于数组名就是数组的首地址。我们可以通过反汇编直观的查看到函数名和函数地…

【转载】Direct3D基础知识

原文&#xff1a;Direct3D基础知识 重新从头开始学习DX,以前太急于求成了,很多基础知识都没掌握就开始写程序了,结果出了问题很难解决.1. D3D体系结构D3D与GDI处与同一层次,区别在于,D3D可以使用HAL(Hardware Abstraction Layer)通过DDI来访问图形硬件,充分发挥硬件性能.…

翻译:程序员数据结构基础:选择正确的数据结构

本文转载自GameDev.net&#xff0c;仅供学习交流。因为刚刚开始学习翻译&#xff0c;难免有些疏漏&#xff0c;如果有哪些地方翻译的不正确&#xff0c;请不吝告知&#xff0c;万分感谢。 原文链接&#xff1a;http://www.gamedev.net/page/resources/_/technical/general-prog…

关于Xcode隐藏打印的logs的方法

https://www.cnblogs.com/jukaiit/p/5881062.html 第一步&#xff1a; 第二步&#xff1a; 第三步&#xff1a; 添加参数&#xff1a; Name &#xff1a;OS_ACTIVITY_MODE Value : disable

指针函数与函数指针的区别

首先它们之间的定义&#xff1a;1、指针函数是指带指针的函数&#xff0c;即本质是一个函数&#xff0c;函数返回类型是某一类型的指针。 类型标识符 *函数名(参数表)int *f(x&#xff0c;y);首先它是一个函数&#xff0c;只不过这个函数的返回值是一个地址值。函数返回值必须用…

数组字典

#import <Foundation/Foundation.h>int main(int argc, const char * argv[]) {autoreleasepool { //把字典放到数组中NSDictionary *dic1{"name":"小明","class":"IOS8","age":"22"};NSDictionary *dic2{&…

C++走向远洋——63(项目二2、两个成员的类模板)

*/ * Copyright (c) 2016&#xff0c;烟台大学计算机与控制工程学院 * All rights reserved. * 文件名&#xff1a;text.cpp * 作者&#xff1a;常轩 * 微信公众号&#xff1a;Worldhello * 完成日期&#xff1a;2016年6月4日 * 版本号&#xff1a;V1.0 * 问题描述&…

iOS 抓包工具 charles工具

在Charles官网下载最新的 安装包 在电脑上安装完成之后&#xff0c;以 注册码 Registered Name: https://zhile.io License Key: 48891cf209c6d32bf4 进行注册即可完成 在手机上面设置代理&#xff1a;输入电脑的网络IP以及端口号 以下为查找的步骤&#xff1a; 在手机上手…

指针数组与数组指针详解

指针数组与数组指针详解 1.什么是指针数组和数组指针&#xff1f; 指针数组&#xff1a;指针数组可以说成是”指针的数组”&#xff0c;首先这个变量是一个数组&#xff0c;其次&#xff0c;”指针”修饰这个数组&#xff0c;意思是说这个数组的所有元素都是指针类型&#xff0…

写一个Android输入法01——最简步骤

本文演示用Android Studio写一个最简单的输入法。界面和交互都很简陋&#xff0c;只为剔肉留骨&#xff0c;彰显写一个Android输入法的要点。 1、打开Android Studio创建项目&#xff0c;该项目和普通APP的不同之处在于它不需要添加任何Activity&#xff1a;我给该输入法命名为…

句柄与指针的区别

句柄实际上是一种指向某种资源的指针&#xff0c;但与指针又有所不同&#xff1a;指针对应着一个数据在内存中的地址&#xff0c;得到了指针就可以自由地修改该数据。 Windows并不希望一般程序修改其内部数据结构&#xff0c;因为这样太不安全。所以Windows给每个使用GlobalAll…

iOS 11 适配

http://blog.csdn.net/st646889325/article/details/79066361 这一个不错的文章

谈谈自己对于Auth2.0的见解

Auth的原理网上有很多&#xff0c;我这里就不在赘述了。 这里有张时序图我个人觉得是比较合理而且直观的&#xff0c;&#xff08;感谢这篇博文&#xff1a;http://justcoding.iteye.com/blog/1950270&#xff09; 参照这个流程&#xff0c;模拟了下部分代码&#xff0c;当然是…