Show和ShowDialog无效

时间:2013-04-21 05:46:15

标签: c# .net multithreading winforms

这不像我以前见过的任何东西 当我调用(new System.Windows.Forms.Form())。ShowDialog()时,表单会显示一毫秒或者其他东西然后消失。
我追踪了这些电话,得到了这个:

  

System.Windows.Forms.Form.Dispose
    System.ComponentModel.Component.Dispose
    System.Windows.Forms.Application.ThreadWindows.Dispose
    System.Windows.Forms.Application.ThreadContext.DisposeThreadWindows
    System.Windows.Forms.Application.ComponentManager.System.Windows.Forms.UnsafeNativeMethods.IMsoComponentManager.FPushMessageLoop
    System.Windows.Forms.Application.ThreadContext.RunMessageLoopInner
    System.Windows.Forms.Application.ThreadContext.RunMessageLoop
    System.Windows.Forms.Form.ShowDialog Esfand.Program.Main C#

我已经尝试过想到解决这个问题的任何事情。

虽然我在尝试显示此表单之前已显示登录表单 我认为登录表单上没有任何特殊内容(通常很无聊,连接到服务器,发送凭据,接收数据)。
我正在使用表单的主线程。我有Messaging和消息循环的经验。我在登录表单中使用了线程。

编辑: 澄清Cody Gray的建议:

这就是我在void Main(string[])中所拥有的:

LoginForm login = new LoginForm ();
login.ShowDialog ();//works
if (login.DialogResult == DialogResult.OK)
{
    MainForm f = new MainForm ();
    f.ShowDialog ();//won't work
}

在新线程中创建和显示MainForm使得所有内容都重新开始工作。但是每个表单上都会出现随机错误,导致此解决方案不够好。

编辑2:
FormClosing事件甚至没有触发。

System.Windows.Forms.Form A;
A = new Form();
A.FormClosing += new FormClosingEventHandler((sender, e) => { System.Diagnostics.Debugger.Break();/*won't work. tried Breakpoints and other stuff too*/ });
A.ShowDialog();

编辑3: HandleDestroyed事件堆栈跟踪:

>   Esfand.exe!Esfand.Program.Main.AnonymousMethod__1(object sender = {System.Windows.Forms.Form}, System.EventArgs e = {System.EventArgs}) Line 50 + 0x6 bytes C#
    System.Windows.Forms.dll!System.Windows.Forms.Control.OnHandleDestroyed(System.EventArgs e) + 0x9e bytes    
    System.Windows.Forms.dll!System.Windows.Forms.Form.OnHandleDestroyed(System.EventArgs e) + 0x13 bytes   
    System.Windows.Forms.dll!System.Windows.Forms.Control.WmDestroy(ref System.Windows.Forms.Message m) + 0x54 bytes    
    System.Windows.Forms.dll!System.Windows.Forms.Control.WndProc(ref System.Windows.Forms.Message m) + 0x547 bytes 
    System.Windows.Forms.dll!System.Windows.Forms.Form.WndProc(ref System.Windows.Forms.Message m) + 0x6d bytes 
    System.Windows.Forms.dll!System.Windows.Forms.NativeWindow.DebuggableCallback(System.IntPtr hWnd, int msg = 2, System.IntPtr wparam, System.IntPtr lparam) + 0x15e bytes    
    [Native to Managed Transition]  
    [Managed to Native Transition]  
    System.Windows.Forms.dll!System.Windows.Forms.NativeWindow.DestroyHandle() + 0xf7 bytes 
    System.Windows.Forms.dll!System.Windows.Forms.Control.DestroyHandle() + 0x3e3 bytes 
    System.Windows.Forms.dll!System.Windows.Forms.Control.Dispose(bool disposing) + 0x347 bytes 
    System.Windows.Forms.dll!System.Windows.Forms.ContainerControl.Dispose(bool disposing) + 0x19 bytes 
    System.Windows.Forms.dll!System.Windows.Forms.Form.Dispose(bool disposing) + 0x26a bytes    
    System.dll!System.ComponentModel.Component.Dispose() + 0x1b bytes   
    System.Windows.Forms.dll!System.Windows.Forms.Application.ThreadWindows.Dispose() + 0xb3 bytes  
    System.Windows.Forms.dll!System.Windows.Forms.Application.ThreadContext.DisposeThreadWindows() + 0x12d bytes    
    System.Windows.Forms.dll!System.Windows.Forms.Application.ComponentManager.System.Windows.Forms.UnsafeNativeMethods.IMsoComponentManager.FPushMessageLoop(System.IntPtr dwComponentID, int reason, int pvLoopData) + 0x58e bytes    
    System.Windows.Forms.dll!System.Windows.Forms.Application.ThreadContext.RunMessageLoopInner(int reason = 4, System.Windows.Forms.ApplicationContext context = {System.Windows.Forms.Application.ModalApplicationContext}) + 0x593 bytes 
    System.Windows.Forms.dll!System.Windows.Forms.Application.ThreadContext.RunMessageLoop(int reason, System.Windows.Forms.ApplicationContext context) + 0x81 bytes    
    System.Windows.Forms.dll!System.Windows.Forms.Form.ShowDialog(System.Windows.Forms.IWin32Window owner) + 0x765 bytes    
    Esfand.exe!Esfand.Program.Main(string[] a = {string[0]}) Line 51 + 0x10 bytes   C#

6 个答案:

答案 0 :(得分:5)

  

这件事使我的程序中的每一个表单都产生了一个独特的错误(例如``无法注册拖放事件处理程序')

这是对代码核心问题的强烈暗示。当您具有AllowDrop属性设置为true的任何控件时,将调用RegisterDragDrop()本机函数。在创建表单的本机窗口时调用它。不幸的是,如果您拥有64位版本的Windows并且强制程序以32位模式运行,那么引发任何异常都是非常糟糕的时间。 this answer的主题。

RegisterDragDrop()失败的原因很少。但是一个。我们已经可以从你的代码片断中告诉你,你一直在修改Program.cs。它的样板版本如下所示:

static class Program {
    /// <summary>
    /// The main entry point for the application.
    /// </summary>
    [STAThread]                // <=== Here!!!
    static void Main() {
        Application.EnableVisualStyles();
        Application.SetCompatibleTextRenderingDefault(false);
        Application.Run(new Form1());
    }
}

我把Big Arrow放在重要的代码行上。 [STAThread]属性在任何GUI程序中都是必不可少的。它告诉CLR它应该初始化COM并将程序的主线程配置为Single Threaded Apartment。公寓是一个非常重要的COM实施细节,其范围有点超出这个答案。如果缺少该属性,则程序的主线程将加入多线程单元MTA。对于非线程安全的代码的恶意地方,例如拖放,剪贴板和shell对话框。

忘记使用该属性可能会导致令人困惑的异常。特别糟糕的是当您的开发机器启动64位版本的Vista或Win7时,Windows版本在关键时刻出现异常时遇到问题,如链接答案中所述。

您的Program.Main()方法的正确版本没有此问题,否则使用建议的做法:

    [STAThread]                // <=== Don't forget this
    static void Main() {
        Application.EnableVisualStyles();
        Application.SetCompatibleTextRenderingDefault(false);
        using (var login = new LoginForm()) {
            if (login.ShowDialog() != DialogResult.OK) return;
        }
        Application.Run(new MainForm());
    }

答案 1 :(得分:1)

尝试检查是否抛出线程异常错误。检查您是否在Application_ThreadException事件中看到任何内容。

static void Main()
{
    Application.ThreadException += new System.Threading.ThreadExceptionEventHandler(Application_ThreadException);
    try
    {
        //Your existing code
    }
    catch (Exception ex)
    {
    }
}

private static void Application_ThreadException(object sender, System.Threading.ThreadExceptionEventArgs e)
{
}

修改 另一种选择是将MainForm显式设置为正在创建的newForm的所有者。

newForm.ShowDialog(MainForm);

我觉得默认情况下所有者已设置为您的“登录”表单已关闭,这是自动关闭新表单

答案 2 :(得分:1)

当事件队列中有WM_QUIT消息时,IMsoComponentManager.FPushMessageLoopP()看起来会调用Application.ThreadContext.DisposeThreadWindows()

您是否在LoginForm的按钮事件处理程序中发布了退出消息?

答案 3 :(得分:0)

尝试将表单定义为类成员。不在函数内部。

答案 4 :(得分:0)

您的Main方法看起来很奇怪。我想你错过了对Application.Run()的电话。由于您不希望应用程序在登录表单关闭后立即退出,因此您可能需要ApplicationContext。 MSDN有一个例子:http://msdn.microsoft.com/en-us/library/ms157901.aspx

另一种可能性是用一个不可见的形式调用Application.Run()作为参数然后显示其他形式(并且在应用程序退出之前永远不会关闭),但在我看来这是一个非常难看的黑客。

答案 5 :(得分:0)

我的理解是表单需要在应用程序上下文中运行。

我继承了一些代码,它们以下列方式从Main上下文中启动多个表单:

var form1 = Form1();
Application.Run(form1);

// form returns, check public method form.ButtonPushed etc
if (form.Button1Pushed)
{
    var form2 = Form2();
    Application.Run(form2);
}

这将成功启动多种表格。

它不是一种非常优雅的做事方式,但它有效...