6

我在显示 FolderBrowserDialog 时随机收到 InvalidCastException,并且许多客户也报告了这一点。

我无法在互联网上找到任何相关的内容。有谁知道这是什么原因/如何解决这个问题?

我的代码:

        using (FolderBrowserDialog fbd = new FolderBrowserDialog())
        {
            fbd.ShowNewFolderButton = false;
            if (fbd.ShowDialog() == DialogResult.OK)

堆栈跟踪:

Error: System.InvalidCastException: 
'Unable to cast object of type 'System.__ComObject' to type 'IMalloc'.'.

    Stack trace:    
at System.Windows.Forms.UnsafeNativeMethods.Shell32.SHGetMalloc(IMalloc[] ppMalloc)
at System.Windows.Forms.FolderBrowserDialog.GetSHMalloc()
at System.Windows.Forms.FolderBrowserDialog.RunDialog(IntPtr hWndOwner)
at System.Windows.Forms.CommonDialog.ShowDialog(IWin32Window owner)
at System.Windows.Forms.CommonDialog.ShowDialog()

编辑:附加信息:只有在 VS2008 调试器中运行时,我才能重现这一点。

当调试器用完时,它在我的 64 位 Windows 7 上很少发生(在 6 个月内发生一次或两次),并且在重新启动后消失。

客户端肯定不会在调试器中运行应用程序,因此它肯定可以在调试器之外重现。

4

3 回答 3

1

这里有几个想法:

据我所知,使用 Reflector.Net 在实际对话框返回后,它会被抛出到 finally 块中。这基本上是您遇到问题的地方:

IntPtr pszPath = IntPtr.Zero;
try
{
    UnsafeNativeMethods.BROWSEINFO lpbi = new UnsafeNativeMethods.BROWSEINFO();
    hglobal = Marshal.AllocHGlobal((int) (260 * Marshal.SystemDefaultCharSize));
    pszPath = Marshal.AllocHGlobal((int) (260 * Marshal.SystemDefaultCharSize));
    ... /*init structure*/
    pidl = UnsafeNativeMethods.Shell32.SHBrowseForFolder(lpbi);
    if (pidl != IntPtr.Zero)
    {
        UnsafeNativeMethods.Shell32.SHGetPathFromIDList(pidl, pszPath);
        ...
    }
}
finally
{
    UnsafeNativeMethods.IMalloc sHMalloc = GetSHMalloc(); /* Boom! */
    sHMalloc.Free(zero);
    ...

如果您根本没有看到上面的异常对话框,则可能掩盖了真正的错误。尝试使用“Break on Exception”运行并禁用 Tools->Debugging->Just my code。try 块中的代码看起来非常基本,他们正在做的最危险的事情是在 shell32.dll 的 SHBrowseForFolder 上 PInvoke 如果它生成“随机”错误,我会感到惊讶。

如果您看到对话框并且只有在关闭时才会收到此错误,那么您可以忽略它,但发生这种情况时会泄漏内存:

    using (FolderBrowserDialog fbd = new FolderBrowserDialog())
    {
        fbd.ShowNewFolderButton = false;
        DialogResult r;
        try { r = fbd.ShowDialog(); }
        catch (InvalidCastException) 
        { r = DialogResult.OK; /* you might check the path first */ }
        if (fbd.ShowDialog() == DialogResult.OK)
            ...

当然,您始终可以自己PInvoke SHBrowseForFolder而不使用对话框类。

于 2010-05-11T16:58:10.720 回答
0

在我的项目中,我遇到了几乎相同的问题(也是 InvalidCastException),只是有时会发生。

它来自一个尚未作为 STAThread 运行的线程。虽然我的 Main 方法被标记为 [STAThread] 属性。

您说,您没有使用单独的线程。但也许你不知道,因为异步委托,它没有显式使用 Thread 类,而是被视为一个。

如果您创建新线程(不管您是使用 ThreadPool 还是异步委托创建它),它们始终是 MTA Threads。因此,您必须自己创建线程并将其作为 STAThread 显式启动。

你可以这样做:

var thread=new Thread( () => method() );
thread.SetApartmentState(ApartmentState.STA);
thread.Start();

我认为您必须朝那个方向挖掘才能找到错误。

于 2010-05-12T14:31:24.397 回答
0

这种症状似乎发生在其他人身上,所以至少你并不孤单;-)

几种可能性:

  1. 您是否在单线程单元中运行它(即在入口点方法上使用 [STAThreadAttribute])?
  2. Windows 中的最大路径长度为 260 个字符。FolderBrowserDialog 使用的初始路径会比这长吗?如果您可以(偶尔)在 VS 调试模式下重现此问题,请尝试将您的解决方案移到文件夹树的更高位置,从而缩短对话框使用的默认文件夹路径。
于 2010-05-11T17:19:18.407 回答