为什么SynchronizationContext.Current为null?

时间:2011-08-16 08:30:21

标签: c# .net winforms multithreading

错误:Object reference not set to an instance of an object.

以下算法有效。 我尝试过,然后将Winform项目移到另一个目录,SynchronizationContext.Currentnull。 为什么呢?

SynchronizationContext uiCtx = SynchronizationContext.Current;  

private void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e)
{
    int[] makeSelfMoves = new int[4];

    lock (replay)
    {
        // count should be more than 2
        foreach (KeyValuePair<int, int[]> item in replay)
        {              
            makeSelfMoves = replay[item.Key];
            codeFile.ExecuteAll(makeSelfMoves[0],
              makeSelfMoves[1], makeSelfMoves[2], makeSelfMoves[3]);

            // i get the error here. uictx is null
            uiCtx.Post(o =>
            {
                PrintPieces(codeFile.PieceState());
            }, null);                               

            System.Threading.Thread.Sleep(1000);
        }
    }
}

2 个答案:

答案 0 :(得分:14)

您的代码严格依赖于类的构造函数的运行时间和位置。在以下情况下,SynchronizationContext.Current将为null:

  • 在您的代码创建Form类的实例或在Main()中调用Application.Run()之前,您的类对象过早创建。当Current成员设置为WindowsFormsSynchronizationContext的实例时,该类知道如何使用消息循环封送调用。通过将对象实例化代码移动到主窗体构造函数来解决此问题。

  • 您的类对象是在主UI线程以外的任何线程上创建的。只有Winforms应用程序中的UI线程才能封送调用。通过使用以下语句向您的类添加构造函数来诊断:

            Console.WriteLine(System.Threading.Thread.CurrentThread.ManagedThreadId);
    

还要将此行添加到Program.cs中的Main()方法中。如果“输出”窗口中的显示值不同,则无效。通过将对象实例化代码再次移动到主窗体构造函数来解决此问题,以便确保它在UI线程上运行。

答案 1 :(得分:0)

我在测试框架中通过依赖注入创建 WinForms 时遇到了这个问题。最初我会在我的构造函数中捕获 SynchronizationContext.Current:

private readonly SynchronizationContext UISyncCtxt;

public MyWinFormViewModel ()
{
    UISyncCtxt = SynchronizationContext.Current;
    ...
}

如果这个 MyWinFormViewModel 是在应用程序已经运行时创建的,那么这工作正常,但这不一定是在测试工具中创建依赖图时的情况。当由测试工具创建时,SynchronizationContext.Current 将为空,并且稍后会出现空引用异常。

我的解决方案是“懒惰地”评估它如下:

    private SynchronizationContext _uisyncctxt;

    private SynchronizationContext UISyncCtxt => 
        _uisyncctxt ??= SynchronizationContext.Current;

当我真正需要上下文(更新表单上的控件)时,它肯定存在(因为表单已被实例化)。

编辑:Peter Duniho 提出了一个关于任意获取同步上下文的有效观点。我的原始答案也使这个类对其依赖项不诚实,因为它依赖于这个上下文,但不通过构造函数或其他可注入方法请求它。由于这个类使用了 DI,所以我添加了一个名为 IUISyncContext 的依赖项,其签名如下:

public interface IUISyncContext
{
    SynchronizationContext UISyncContext { get; }
}

...以及我的视图模型的构造函数:

private readonly SynchronizationContext UISyncCtxt;

public MyWinFormViewModel (IUISyncContext syncContext)
{        
    UISyncCtxt = syncContext.UISyncContext;
    ...
}

感谢您的反馈,彼得。