0

由于我的英语不好,我简单地解释我的问题并在此处粘贴代码片段来描述问题。

问题是我们的 winForm 应用程序中的多线程问题。我将逻辑简化为以下代码示例。

在测试代​​码中,mainForm中有1个mainForm Form1和一个名为“开始”的按钮。当用户单击该按钮时,将从 2 个后台线程显示两个表单 form2 和 form3。form2 关闭后,Form1 将被触发关闭。但是这里显示了form3,所以我需要用户自己关闭form3。所以我处理了 form.Closing 事件并使用 Application.DoEvents() 让用户关闭 form3。它看起来很有效。但实际上,form3 可以接受用户的操作,但 form3 不会按预期关闭。

请说明为什么form3不能在这里关闭以及如何修改代码使用户的关闭操作起作用。

using System;
using System.ComponentModel;
using System.Threading;
using System.Windows.Forms;

namespace CloseFormFromMainThread
{
public partial class Form1 : Form
{
    private Form2 _form2;
    private Form2 _form3;
    private SynchronizationContext _synchronizationContext;

    public Form1()
    {
        InitializeComponent();
        Closing += Form1Closing;
    }

    void Form1Closing(object sender, CancelEventArgs e)
    {
        while (_form3 != null)
        {
            Application.DoEvents();
            Thread.Sleep(100);
        }
    }

    private void ButtonStartClick(object sender, EventArgs e)
    {
        var thread = new Thread(StartForm3);
        thread.Start();

        var thread2 = new Thread(StartForm2);
        thread2.Start();
    }

    private void StartForm3()
    {
        Thread.Sleep(200);
        var action = new Action(() =>
                                    {
                                        _form3 = new Form2();
                                        _form3.Text = "form 3";
                                        _form3.ShowDialog();
                                        _form3 = null;
                                    });
        ExecuteActionInUiThread(action);
    }

    private void Form1Load(object sender, EventArgs e)
    {
        _synchronizationContext = SynchronizationContext.Current;
    }

    private void StartForm2()
    {
        Thread.Sleep(500);
        var action = new Action(() =>
        {
            _form2 = new Form2();
            _form2.Text = "form 2";
            _form2.ShowDialog();

            Close();
        });
        ExecuteActionInUiThread(action);
    }

    private void ExecuteActionInUiThread(Action action)
    {
        var sendOrPostCallback = new SendOrPostCallback(o => action());
        _synchronizationContext.Send(sendOrPostCallback, null);
    }
}
}
4

2 回答 2

2

第一个建议:不要使用 Application.DoEvents()。曾经。每当您认为需要它时,您的代码流中就有一个概念性问题,您应该首先解决它。我猜你的代码只是在创建一个死锁,因为它会等待 OnClosing 回调返回,然后才能处理更多事件(比如关闭另一个表单)。

于 2014-02-12T09:51:12.340 回答
0

当我检查 Form.Close() 的源代码时,对于模态对话框,只引发了 FormClosing 事件,没有引发 Closed 事件。没有发送信号让 form3.ShowDialog 继续。所以我认为只使用主线程和 Application.DoEvents 不能让代码继续。然后是主线程中的软死锁。

目前,我使用另一个线程进行检查 (_form3 != null) 并让主线程执行 _form3.ShowDialog() 逻辑。以下是我关于 Form1Closing 的代码。

    private bool _isFormClosing;
    void Form1Closing(object sender, CancelEventArgs e)
    {
        if (_form3 == null) return;

        e.Cancel = true;
        if (!_isFormClosing)
        {
            _isFormClosing = true;
            Task.Factory.StartNew((() =>
            {
                while (_form3 != null)
                {
                    Thread.Sleep(50);
                }

                ExecuteActionInUiThread(Close);
            }));
        }
    }
于 2014-02-15T03:51:14.067 回答