关于C#调用C++动态库的文章很多,调用动态库中回调函数的方法也不在少数。但大多数调用回调函数的方法依然保留了C++的语法特点。

  比如有一段C++的回调函数代码,为了表达它的意思,我把注释也粘贴了进来:

    /*********************************************************************************************************
   ** Function name:     epcBuzzerAsyncOn
   ** Descriptions:      本函数使蜂鸣器蜂鸣指定时间
   ** input parameters:        dwOntime    蜂鸣器持续蜂鸣的时间(ms),其中0表示一直蜂鸣
   ** output parameters:    无
   ** Returned value:          TRUE:成功;FALSE:失败
   ** Note:                    本函数以异步方式执行,不会被阻塞
   *********************************************************************************************************/
EPCBUZZERLIB_API BOOL epcBuzzerAsyncOn (DWORD dwOntime);   /*********************************************************************************************************
** Function name: epcBuzzerSetCallBackFunc
** Descriptions: 设置回调函数指针,当异步的蜂鸣器操作任务完成后,会调用该回调函数通知用户程序。
** input parameters: lpfnNotify 回调函数指针,如果是NULL,则表示不需要通知用户程序。函数类
** 型为void (*lpfnNotify)(BOOL bResult),bResult为执行结果,
** TRUE表示执行成功,FALSE表示执行失败
** output parameters: 无
** Returned value: 无
*********************************************************************************************************/
EPCBUZZERLIB_API VOID epcBuzzerSetCallBackFunc( void (*lpfnNotify)(BOOL));

 

  一般的文章都会教你转换成如下代码,这些方法都属于Buzzer类:

     /// <summary>
/// 使蜂鸣器蜂鸣指定时间,异步执行,不会阻塞当前线程.
/// </summary>
/// <param name="onTime">蜂鸣时间(ms).</param>
/// <returns>true:蜂鸣成功,false:蜂鸣失败</returns>
[System.Runtime.InteropServices.DllImport("Lib\\epcBuzzerLib.dll", SetLastError = true)]
private static extern bool epcBuzzerAsyncOn(int onTime);      /// <summary>
/// 先申明一个委托,该委托定义了蜂鸣完成回调函数的指针和数据类型
/// </summary>
/// <param name="result">if set to <c>true</c> [result].</param>
public delegate void BuzzerHandler(bool result); /// <summary>
/// 再设置蜂鸣完成回调函数.
/// </summary>
/// <param name="handler">The handler.</param>
[System.Runtime.InteropServices.DllImport("Lib\\epcBuzzerLib.dll", SetLastError = true)]
private static extern void epcBuzzerSetCallBackFunc(BuzzerHandler handler);

  然后在主程序中如此调用:

//先定义一个满足委托类型的函数
public void BuzzerNotify(bool result)
{
   //在这里编写蜂鸣完成后你想要执行的代码...我这里根据result随便写几行。
if(result)
{
  MessageBox.Show("蜂鸣成功");
}
else
{
  MessageBox.Show("蜂鸣失败");
}
} //在按钮事件中调用回调函数
private void onButton_Click(object sender, EventArgs e)
{
  Buzzer.epcBuzzerCallBackFunc(this.BuzzerNotify);
  Buzzer.epcBuzzerAsyncOn();
}

这样,在蜂鸣器蜂鸣3000ms之后,就会调用BuzzerNotify,并将一个bool结果传进去。然后你可以在BuzzerNotify中做你想做的事。

  虽然这样也能够解决问题,但是我总感觉很别扭,总觉得我的代码里有C++风格,这让有代码洁癖的我感觉到浑身不自在。所以,我决定重构一下这段代码,让它符合C#的风格。

  重构的思想是这样的,虽然我不太能完全理解回调函数这个东西,但是我能隐约感觉到它就是一个事件触发机制,比如上面那段代码,就可以理解为当蜂鸣完成后,就会通知BuzzerNotify,让它开始执行。只不过BuzzerNotiy是在主函数中定义的一个函数,它不能像事件那样可以注册,你所有想在蜂鸣器完成后做的工作,都必须写在这个函数里面。很明显,如此糟糕的扩展性,完全不是C#的风格。所以,我要做的,就是将BuzzerNotify变成Buzzer类的一个事件,让它可以使用+=这样的神器!

  其实做法也是蛮简单的,我把代码一贴出来大家立马就能明白了。以下是Buzzer类的代码

        /// <summary>
/// 申明蜂鸣完成回调函数指针和数据类型
/// </summary>
/// <param name="result">if set to <c>true</c> [result].</param>
public delegate void BuzzerHandler(bool result); /// <summary>
/// 蜂鸣完成时触发该事件.
/// </summary>
public static event BuzzerHandler BuzzingComplete; /// <summary>
/// 使蜂鸣器蜂鸣指定时间,异步执行,不会阻塞当前线程.
/// </summary>
/// <param name="onTime">蜂鸣时间(ms).</param>
/// <returns>true:蜂鸣成功,false:蜂鸣失败</returns>
public static bool TurnOn(int time)
{
Buzzer.BuzzingCompleteSetCallBack();
return Buzzer.epcBuzzerAsyncOn(time);
} /// <summary>
/// 使蜂鸣器蜂鸣指定时间,异步执行,不会阻塞当前线程.
/// </summary>
/// <param name="onTime">蜂鸣时间(ms).</param>
/// <returns>true:蜂鸣成功,false:蜂鸣失败</returns>
[System.Runtime.InteropServices.DllImport("Lib\\epcBuzzerLib.dll", SetLastError = true)]
private static extern bool epcBuzzerAsyncOn(int onTime); /// <summary>
/// 定义蜂鸣完成回调函数,用来处理事件.这个方法比较关键,它相当于把之前的BuzzerNotify,只不过把它封装进Buzzer类了。
/// </summary>
/// <param name="result">if set to <c>true</c> [result].</param>
private static void Event(bool result)
{
if (BuzzingComplete != null)
{
BuzzingComplete(result);
}
} /// <summary>
/// 蜂鸣完成回调函数.
/// </summary>
private static void BuzzingCompleteSetCallBack()
{
Buzzer.epcBuzzerSetCallBackFunc(Event);
} /// <summary>
/// 设置蜂鸣完成回调函数.
/// </summary>
/// <param name="handler">The handler.</param>
[System.Runtime.InteropServices.DllImport("Lib\\epcBuzzerLib.dll", SetLastError = true)]
private static extern void epcBuzzerSetCallBackFunc(BuzzerHandler handler);

虽然多了很多代码,但为了面向对象,这绝对是值得的!
在主函数中可以如此调用:

private void FirstThing(bool result)
{
MessageBox.Show("Do my first thing");
} private void SecondThing(bool result)
{
MessageBox.Show("Do my second thing");
} private void onButton_Click(object sender, EventArgs e)
{
Buzzer.BuzzingComplete += this.FirstThing;
Buzzer.BuzzingComplete += this.SecondThing;
Buzzer.TurnOn();
}

如此一来,主程序中的代码看上去就舒服多了。值得一提的是,如果你连续点几次这个OKButton的话,这两个方法会被重复注册。

C#将C++动态库的回调函数封装成事件的更多相关文章

  1. Golang编写动态库实现回调函数

    Golang编写动态库实现回调函数 我们现在要做一个动态库,但是C++实在是比较难,于是就想能不能用更简单的golang来实现,golang也就是最近的版本才支持编译成动态库,在网上也没找到可用的案例 ...

  2. Golang调用windows下的dll动态库中的函数

    Golang调用windows下的dll动态库中的函数 使用syscall调用. package main import ( "fmt" "syscall" & ...

  3. windows下查看静态库和动态库的导出函数

    在window下查看动态库的导出函数可以用vs自带的Depends工具: 查看静态库的信息要用命令行来实现: dumpbin   /LINKERMEMBER   Test.lib   >   1 ...

  4. Golang调用windows下的dll动态库中的函数 Golang 编译成 DLL 文件

    Golang调用windows下的dll动态库中的函数 package main import ( "fmt" "syscall" "time&quo ...

  5. 回调函数与DOM事件

    原文:http://dean.edwards.name/weblog/2009/03/callbacks-vs-events/ 先看如下代码: document.addEventListener(&q ...

  6. 动态库(.so)隐藏函数名

    一.偶遇 error: undefined reference to  xxx 问题 尝试封装通用的接口到一个private.so,然后供客户端使用,private.so编译出来后由sample.cp ...

  7. c++ 回调函数封装

    std::function<void(int a,int b)> ha; //函数封装  当成参数用callback  std::bind(&fun1,this,std::plac ...

  8. PCL中将回调函数封装到类中

    这是类中的声明 private://点云回调函数 NuClearTask_MyPointCloudHandle //点云选择 static void ps_callback(const pcl::vi ...

  9. java将类和函数封装成jar,然后在别的项目中使用这个jar包

    本来想用idea安装的,不过用maven生成后发现jar有20,30M肯定不对,后来还是用eclipse生成了,方便很多 环境: eclipse luna,jdk1.8_112 1.生成jar包,首先 ...

随机推荐

  1. C/C++中产生随机数(rand,srand用法)

    计算机的随机数都是由伪随机数,即是由小M多项式序列生成的,其中产生每个小序列都有一个初始值,即随机种子.(注意: 小M多项式序列的周期是65535,即每次利用一个随机种子生成的随机数的周期是65535 ...

  2. 【linux】关机重启命令

    shutdown: [参数][时间] -h:关机 -r:重启 -c:取消上一次关机或重启 [root@paulinux ~]# shutdown -h now ##马上重启 [root@paulinu ...

  3. 【Spring-AOP-学习笔记-7】@Around增强处理简单示例

    阅读目录 简单介绍 章节1:项目结构 章节2:定义切面类.连接点注解类 章节3:为待增强的方法--添加注解声明 章节4:AspectJ配置文件 章节5:测试类xxx 章节6:测试结果 Around 增 ...

  4. nginx按天切割日志

    原文链接:http://www.cnblogs.com/benio/archive/2010/10/13/1849935.html  本文只节选部分内容 Nginx自己没有日志分割的功能,一旦时间过长 ...

  5. android学习笔记39——使用原始资源

    原始资源 android中没有专门提供管理支持的类型文件,都被称为原始资源.例如:声音资源... android原始资源存放位置: 1.res/raw,android SDK会处理该目录下的原始资源, ...

  6. idea系列新版注册模式

    http://idea.qinxi1992.cn/ 楼上被列入黑名单,用 http://114.215.133.70:41017/

  7. C#中WebService 的 Timer定时器过段时间后自动停止运行

    我用.net做的一个Timer定时器,定时获取短信并给予回复,但大概过了十几个小时以后,Timer定时器会自动停止,再发送短信就不能收到回复,需要在服务器中重新运行定时器才可以,请教各位! 我是在.n ...

  8. select,epool,pool解释

    内容主要来自搜狗实验室技术交流文档, 编写链接数巨大的高负载服务器程序时,经典的多线程模式和select模式都不再适合了.应该采用epool/kqueue/dev_pool来捕获IO事件. ----- ...

  9. AP_AP系列 - 供应商管理(案例)

    2014-07-03 Created By BaoXinjian

  10. tcpdump学习

    #直接启动tcpdump将监视第一个网络接口上所有流过的数据包 -n不解析地址到nametcpdump -n #监视指定网络接口的数据包,不指定则为 eth0tcpdump -i eth1 #监视指定 ...