0

我有一个 CE6.0R3 系统,它使用 .NetCF3.5 并使用 P/Invoke 来连接几个本机 C++ dll。

这些设备偶尔会崩溃,有时我们会看到一个弹出窗口,上面写着“应用程序 myc#app.exe 遇到严重错误,必须关闭”。

我们所做的其中一件事是将 C# 表单的句柄传递给本地 C++ 应用程序,该应用程序使用 DirectShow 在 C# 表单上呈现视频。

我一直在研究它,并发现了一些关于使用 GCHandle 固定托管对象以使其不会移动的讨论。

我添加了诊断程序来尝试检测这是否是问题,并添加了 GCHandle.Alloc()s 来固定对象。出于某种原因,对象似乎在内存中移动,即使它们应该被固定。

这是我的代码的一部分:

Diag.Putline("_videoplay create");
_videoplay = new Form();

Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " before pinning Handle");
GCHandle gch = GCHandle.Alloc(_videoplay.Handle, GCHandleType.Pinned);
Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " before pinning Form");
GCHandle gch_videoplay = GCHandle.Alloc(_videoplay, GCHandleType.Pinned);    // Pin the _videoplay object instance so it won't get moved by GC
Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " after pinning");

Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " before .Location");
_videoplay.Location = new Point(x, y);
Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " before .Size");
_videoplay.Size = new Size(w, h);
Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " before .BackColor");
_videoplay.BackColor = bgColor;
Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " before .FormBorderStyle");
_videoplay.FormBorderStyle = FormBorderStyle.None;
Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " before .Owner");
_videoplay.Owner = _mediaform;
Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " before .Show()");
_videoplay.Show();
Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " after .Show()");

Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " before directshow_connect");
Diag.Putline("directshow_connect");

// P/Invoke to our native C++ application based on this code: http://msdn.microsoft.com/en-us/library/ms899491.aspx
// Pass the handle ot our form that we want to display the video on
// the directshow HWND will be a WS_CHILD of _videoplay
directshow_connect(_videoplay.Handle);
Diag.Putline("_videoplay.Handle = " + _videoplay.Handle + " after directshow_connect");

和输出:

14:59:37| _videoplay create
14:59:37| _videoplay.Handle = 1879191552 before pinning Handle
14:59:37| _videoplay.Handle = 1879191552 before pinning Form
14:59:37| _videoplay.Handle = 1879191552 after pinning
14:59:37| _videoplay.Handle = 1879191552 before .Location
14:59:37| _videoplay.Handle = 1879191552 before .Size
14:59:37| _videoplay.Handle = 1879191552 before .BackColor
14:59:37| _videoplay.Handle = 1879191552 before .FormBorderStyle
14:59:37| _videoplay.Handle = 1879191776 before .Owner
14:59:37| _videoplay.Handle = 1879192000 before .Show()
14:59:37| _videoplay.Handle = 1879192000 after .Show()
14:59:37| _videoplay.Handle = 1879192000 before directshow_connect
14:59:37| directshow_connect
14:59:39| _videoplay.Handle = 1879192000 after directshow_connect
14:59:41| _videoplay.Handle = 1879193248 (_TickTockThreadProc)
14:59:41| _videoplay.Handle = 1879193248 (_TickTockThreadProc)
14:59:41| _videoplay.Handle = 1879193248 (UpdateTimer_Tick)
14:59:41| _videoplay.Handle = 1879193248 (UpdateTimer_Tick)
14:59:41| _videoplay.Handle = 1879193248 (_TickTockThreadProc)
14:59:42| _videoplay.Handle = 1879193248 (_TickTockThreadProc)

为什么手柄即使被固定也会改变?

4

1 回答 1

2

您误解了 GCHandle 的工作原理和使用方式。

首先,GCHandle 只能用于 blittable 类型,因此您不能固定 Form 本身。你正在做的是固定句柄,这基本上是在你的代码中对 GC 说“不要移动你在内存中拥有表单地址的位置”。意味着Handle本身的存放位置不能移动。没有什么可以阻止 Form 移动,从而阻止 Handle 持有的

句柄正在更改有点奇怪,我怀疑一旦创建了表单本身,本机表单句柄本身就无法更改。这让我觉得你有一个伪句柄。如果是这种情况,在本机调用中使用它应该不起作用。

我并不坚信这个错误是一个动作,因为我以前从未见过这样的行为——我更倾向于认为这是尝试使用已处置表单的句柄,但你一直在进行调试和可能会有更好的感觉。

无论如何,如果您确实认为失败是由于句柄更改引起的,那么最好的解决方法是使用 P/Invoke(到 CreateWindowEx)来创建容器表单本身。GC 不能移动它,因为它对它一无所知。这将消除压实作为罪魁祸首(或解决问题)。

于 2013-05-09T23:35:17.283 回答