日期:2014-05-20  浏览次数:20948 次

如何在非托管C++中调用托管C#中的回调函数
大家好,我刚刚接触.net,下面是msdn关于回调函数的解释
http://msdn.microsoft.com/zh-cn/library/843s5s5x(v=vs.100).aspx
问题是如何在非托管C++中调用托管C#中的回调函数,文章中没有说。
为了防止垃圾回收,我用HandleRef盯住了回调函数,当C#程序执行了
CallBack myCallBack = new CallBack(EnumReportApp.Report);
IntPtr ptr = Marshal.GetFunctionPointerForDelegate(myCallBack);
HandleRef hr = new HandleRef(myCallBack, ptr);
EnumWindows(hr, 0);
之后,myCallBack的指针被传递到了C++,假设指针的形参名是pCallBack,在C++的DLL中直接执行
pCallBack();
然后程序抛出了异常:“访问了受保护的内存区域”,继续运行抛出“回调函数被垃圾回收”。
我觉得应该是C++调用C#的函数指针的方法不对,请大家看看问题出在哪里?


------解决方案--------------------
 全部折叠全部展开      代码:全部 代码:多个 代码:Visual Basic 代码:C# 代码:Visual C++ 代码:J# 代码:JScript  
Visual Basic
C#
Visual C++
J#
JScript
.NET Framework 开发人员指南 
Callback 示例 
请参见  发送反馈意见 
 

该示例说明如何将委托传递给需要函数指针的非托管函数。委托是可以容纳对方法的引用的类,并且等效于类型安全函数指针或回调函数。 

注意: 
当您在调用内部使用委托时,公共语言运行库将在该调用的持续时间内防止对委托执行垃圾回收。但是,如果非托管函数存储该委托以供在该调用完成后使用,则您必须手动防止进行垃圾回收,直到非托管函数完成对该委托的使用为止。有关更多信息,请参见 HandleRef 示例和 GCHandle 示例。
 

Callback 示例使用以下非托管函数(这里同时显示其原始函数声明): 

从 PinvokeLib.dll 导出的 TestCallBack。 

  复制代码 
void TestCallBack(FPTR pf, int value);
 

从 PinvokeLib.dll 导出的 TestCallBack2。 

  复制代码 
void TestCallBack2(FPTR2 pf2, char* value);
 

PinvokeLib.dll 是一个自定义非托管库,它包含前面列出的函数的实现。

在该示例中,LibWrap 类包含 TestCallBack 和 TestCallBack2 方法的托管原型。这两个方法都将委托作为参数传递给回调函数。该委托的签名必须与它所引用的方法的签名相匹配。例如,FPtr 和 FPtr2 委托的签名与 DoSomething 和 DoSomething2 方法的签名相同。

下面的代码示例的源代码由 .NET Framework 平台调用技术示例 提供。

声明原型
Visual Basic  复制代码 
Public Delegate Function FPtr( ByVal value As Integer ) As Boolean
Public Delegate Function FPtr2( ByVal value As String ) As Boolean

Public Class LibWrap
   ' Declares managed prototypes for unmanaged functions.
   Declare Sub TestCallBack Lib "..\LIB\PinvokeLib.dll" ( ByVal cb _
      As FPtr, ByVal value As Integer )
   Declare Sub TestCallBack2 Lib "..\LIB\PinvokeLib.dll" ( ByVal cb2 _
      As FPtr2, ByVal value As String )
End Class 'LibWrap

 
C#  复制代码 
public delegate bool FPtr( int value );
public delegate bool FPtr2( String value );

public class LibWrap
{// Declares managed prototypes for unmanaged functions.
   [ DllImport( "..\\LIB\\PinvokeLib.dll" )]
   public static extern void TestCallBack( FPtr cb, int value );   
   [ DllImport( "..\\LIB\\PinvokeLib.dll" )]
   public static extern void TestCallBack2( FPtr2 cb2, String value );   
}
 

调用函数
Visual Basic  复制代码 
Public Class App