为什么在单击ToolStrip按钮两次时抛出NullReferenceException – openFileDialog.showDialog()?
我创建了一个干净的WindowsFormsApplication解决方案,在主窗体上添加了一个ToolStrip
,并在其上放置了一个按钮。 我还添加了一个OpenFileDialog
,因此ToolStripButton
的Click
事件如下所示:
private void toolStripButton1_Click(object sender, EventArgs e) { openFileDialog1.ShowDialog(); }
我没有改变任何其他属性或事件。
有趣的是,当我双击ToolStripButton
(第二次单击必须非常快,在对话框打开之前),然后取消两个对话框(或选择一个文件,它并不重要),然后单击客户端主窗体的区域, NullReferenceException
崩溃应用程序(post末尾附加的错误详细信息)。 请注意, Click
事件是在DoubleClick
未执行时实施的 。
更奇怪的是,当OpenFileDialog
被任何用户实现的表单替换时, ToolStripButton
阻止被点击两次 。
我在Windows 7 Professional (来自MSDNAA)上使用VS2008和.NET3.5进行最新更新。 我没有在VS中更改很多选项(只有fontsize,工作区文件夹和行编号)。
有谁知道如何解决这个问题? 它在我的机器上是100%可复制的,是否也在其他机器上?
我能想到的一个解决方案是在调用OpenFileDialog.ShowDialog()
然后再启用按钮之前禁用该按钮(但这并不好)。 还有其他想法吗?
现在承诺的错误细节:
System.NullReferenceException未处理
Message =“对象引用未设置为对象的实例。”
来源= “System.Windows.Forms的”
堆栈跟踪:
在System.Windows.Forms.NativeWindow.WindowClass.Callback(IntPtr hWnd,Int32 msg,IntPtr wparam,IntPtr lparam)
在System.Windows.Forms.UnsafeNativeMethods.PeekMessage(MSG&msg,HandleRef hwnd,Int32 msgMin,Int32 msgMax,Int32 remove)
在System.Windows.Forms.Application.ComponentManager.System.Windows.Forms.UnsafeNativeMethods.IMsoComponentManager.FPushMessageLoop(Int32 dwComponentID,Int32 reason,Int32 pvLoopData)
在System.Windows.Forms.Application.ThreadContext.RunMessageLoopInner(Int32 reason,ApplicationContext context)
在System.Windows.Forms.Application.ThreadContext.RunMessageLoop(Int32 reason,ApplicationContext context)
在System.Windows.Forms.Application.Run(Form mainForm)
在WindowsFormsApplication1.Program.Main()w C:\ Users \ Marchewek \ Desktop \ Workspaces \ VisualStudio \ WindowsFormsApplication1 \ Program.cs:第20行
在System.AppDomain._nExecuteAssembly(程序集程序集,String [] args)
在System.AppDomain.ExecuteAssembly(String assemblyFile,Evidence assemblySecurity,String [] args)
在Microsoft.VisualStudio.HostingProcess.HostProc.RunUsersAssembly()
在System.Threading.ThreadHelper.ThreadStart_Context(对象状态)
在System.Threading.ExecutionContext.Run(ExecutionContext executionContext,ContextCallback回调,对象状态)
在System.Threading.ThreadHelper.ThreadStart()
的InnerException:
我能够在Windows 7机器上复制这样的东西 – 我没有exception,但我的表格将不再重绘。 可能是因为我没有在win 7框中的调试器中运行exception被吞下。
这不会发生在我的XP机器上。 这只发生在我第一次打开对话框时使用toolStripButton并双击它时。 如果我正常打开对话框然后先将其关闭,则双击不会打开对话框两次。
我怀疑这里发生的事情类似于竞争条件 – 框架开发人员从未期望他们的代码可以输入两次,但是由于新的回调进入消息循环而发生了这种情况。 那么为什么会发生这种情况 – 对我来说看起来像个错误。
我找到了一个非常简单的解决方法来阻止它发生 – 启用toolStripButton的DoubleClickEnabled属性。 您不必实现双击处理程序 – 它只需单击一次即可将双击视为一次,并且一切正常。
我会这样处理:
public Form1() { InitializeComponent(); // This is a workaround for a framework bug // see blah blah toolStripButton1.DoubleClickEnabled = true; }
下次升级框架时,可以尝试将其删除。
尼尔
ShowDialog方法是模态的。 通常在调用ShowDialog之后,OpenFileDialog具有应用程序的独占UI焦点,直到对话框关闭。
在框架有机会为OpenFileDialog提供独占的UI焦点之前,通过快速双击您再次调用ShowDialog的按钮。 这使您的应用程序处于无效状态。
虽然这不应该是可能的,但显然它是,现在这是你的问题。 订阅按钮的Click事件并在第一次点击后禁用按钮; 或者从Designer中删除OpenFileDialog并在Click处理程序中以编程方式创建它。 如果您执行后者,请将其包装在using块中以确保垃圾回收:
private void toolStripButton1_Click(object sender, EventArgs e) { using(var OFD = new OpenFileDialog()) { OFD.ShowDialog(); } }
屏幕分辨率可能是导致框架错误的原因之一。 在我的情况下,我改变了我的屏幕分辨率,并没有在该分辨率中产生问题,当我转回到推荐的分辨率时,我遇到了问题。