1

我创建了一个非常简单的 winform 应用程序,它有一个表单和一个按钮。它的点击事件如下所示。如果单击该按钮,应用程序将按预期运行,即 ExecTasks 退出。如果将 lock 语句替换为 lock(this),ExecTasks 将不会退出,并且调试器会在 Exec 中的代码处显示处于睡眠/等待/加入状态的线程。问题为什么选择锁定对象会改变这种行为 - 为什么表单实例不是一个有效的选择?

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace TestDeadlock
{
public partial class Form1 : Form
{
    private object _lock = new object();
    public Form1()
    {
        InitializeComponent();
    }

    private void Button1Click(object sender, EventArgs e)
    {
        Task.Factory.StartNew(ExecTasks);
    }

    private void ExecTasks()
    {
        lock (_lock) /* replace by lock(this) to see the blocked behavior */
        {
            var taskList = new List<Task>();
            for (var i = 0; i < 2; ++i)
            {
                taskList.Add(Task.Factory.StartNew(Exec));
            }
            Task.WaitAll(taskList.ToArray());
        }
    }

    private void Exec()
    {
        Invoke((Action)delegate{});
    }
}
}
4

1 回答 1

2

我看到了这种阻塞行为。调用 Invoke 方法时,会调用 Control.FindMarshalingControl 方法,实现如下:

private Control FindMarshalingControl()
{
    lock (this)
    {
        Control parentInternal = this;
        ....
    }
}

这是关于类似问题的链接:

Control.BeginInvoke 也将被阻止

于 2013-01-08T08:38:54.020 回答