如何在非托管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