错误:Object reference not set to an instance of an object.
以下算法有效。
我尝试过,然后将Winform
项目移到另一个目录,SynchronizationContext.Current
为null
。
为什么呢?
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);
}
}
}
答案 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;
...
}
感谢您的反馈,彼得。