c# – 如何在Win32本机函数中使用Marshal.ReleaseComObject

前端之家收集整理的这篇文章主要介绍了c# – 如何在Win32本机函数中使用Marshal.ReleaseComObject前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我正在玩这两个本机win32函数
  1. [DllImport( "oleacc.dll" )]
  2. public static extern int AccessibleObjectFromWindow(
  3. IntPtr hwnd,int dwObjectID,ref Guid refID,ref Accessibility.IAccessible ppvObject );
  4. [DllImport( "oleacc.dll" )]
  5. public static extern uint AccessibleChildren(
  6. Accessibility.IAccessible paccContainer,int iChildStart,int cChildren,[Out] object[] rgvarChildren,out int pcObtained );

而且我很难搞清楚我是否应该/需要在任何返回的对象上调用Marshal.ReleaseComObject.我很感激有人可以开导我这个话题!这是一个示例用法

  1. Accessibility.IAccessible test(
  2. int hWnd,string accessName )
  3. {
  4. Guid guidCOM = new Guid( 0x618736E0,0x3C3D,0x11CF,0x81,0xC,0x0,0xAA,0x38,0x9B,0x71 );
  5. Accessibility.IAccessible a = null;
  6.  
  7. AccessibleObjectFromWindow(
  8. new IntPtr( hWnd ),-4,ref guidCOM,ref a );
  9. object[] children = new object[a.accChildCount];
  10. int z;
  11. AccessibleChildren( a,children.Length,children,out z );
  12. foreach ( Accessibility.IAccessible a2 in children )
  13. try
  14. {
  15. if ( a2.get_accName( 0 ) == accessName )
  16. return a2;
  17. }
  18. catch
  19. {
  20. }
  21. return null;
  22. }

解决方法

Marshal.ReleaseComObject是一个非常有用的解决方法,如果你真的需要让GC立即应用发布,而不是等待GC运行.除非你真的需要,否则最好不要使用它,因为它倾向于接管你的代码而你必须在每个地方都应用它,包括你创建隐式引用的地方.

我怀疑.Net中COM互操作的最佳方法是围绕您正在使用的方法编写强类型包装类.这样就不可能引入隐式引用,它将简化代码并确保GC更容易访问接口.我的一些牛人通过使用这种方法报告了更好的记忆行为 – 当GC运行时,对象被及时释放,而不会留下任何神秘的参考.

一般来说,.Net代码中引用的所有COM对象都以RCW对象的形式实现,该对象保存真正的COM引用 – 对ReleaseComObject的调用强制RCW将引用减少一个计数.这样,它就像在实际的IUnkown实例上调用Release一样.可能在上面的示例中,没有必要拨打电话.

猜你在找的C#相关文章