加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

c# – 当我访问它时,为什么我的初始化静态属性为null?

发布时间:2020-12-15 22:48:16 所属栏目:百科 来源:网络整理
导读:我有以下代码(从实际代码中提取) public static class AssemblyLogger { public static LazyWindow Window { get; } = new LazyWindow(NewWindowHandler); public static IScheduler Scheduler = new DispatcherScheduler( Window.Value.Dispatcher );} 当我
我有以下代码(从实际代码中提取)

public static class AssemblyLogger {

    public static Lazy<Window> Window { get; } = new Lazy<Window>(NewWindowHandler);

    public static IScheduler Scheduler =>
        new DispatcherScheduler( Window.Value.Dispatcher );

}

当我调用Scheduler时,我得到一个NullReferenceException.我看到调试器停止了

enter image description here

据我所知,这应该是不可能的.窗口是静态初始化和只读的,因此任何对它的进一步访问应该只是get,它永远不应该为null.

我已经设置了一个针对初始化器的断点,但它永远不会被击中

enter image description here

我也尝试过静态只读字段,但仍然存在同样的问题

public static readonly Lazy<Window> Window  = new Lazy<Window>(NewWindowHandler);

是否可能对静态初始化有竞争条件?

到目前为止尝试制作MCVE并不成功.以下规范测试我的最小可用代码和真实代码. MCVE通过:(而真正的一个失败.我错过了一些背景,需要进一步的工作来隔离问题.

public class Tester
{
    public static class AssemblyLoggerMCVE
    {

        public static Lazy<Window> Window { get; } = new Lazy<Window>(NewWindowHandler);

        private static Window NewWindowHandler() => new Window();

        public static IScheduler Scheduler =>
            new DispatcherScheduler(Window.Value.Dispatcher);

    }

    /// This passes
    [StaFact]
    public void AssemblyLoggerMCVEShouldWork()
    {
        AssemblyLoggerMCVE.Scheduler.Should().NotBeNull();

    }

    /// This fails
    [StaFact]
    public void AssemblyLoggerShouldWork()
    {
        AssemblyLogger.Scheduler.Should().NotBeNull();

    }
}

解决方法

错误可以通过以下测试用例清楚地复制,该测试用例具有比所建议的问题更多的上下文.

public class Tester
{
    public class AssemblyLoggerControlModel
    {
        public static IScheduler S = AssemblyLoggerMCVE.Scheduler;
    }

    public class AssemblyLogger
    {

        public static AssemblyLoggerControlModel ModelInstance { get; } = 
            new AssemblyLoggerControlModel();

        public static readonly Lazy<Window> Window =
            new Lazy<Window>(NewWindowHandler);

        private static Window NewWindowHandler() => new Window();

        public static IScheduler Scheduler => 
            new DispatcherScheduler(Window.Value.Dispatcher);

    }


}

静态初始化器之间存在循环依赖关系.静态属性的静态初始化AssemblyLogger.ModelInstance导致静态属性AssemblyLoggerControlModel.S被初始化,然后尝试调用静态AssemblyLogger.Scheduler方法,该方法又试图访问Window,因为AssemblyLogger的静态初始化未完成仍为空.

基本上我的代码是纯粹的邪恶,最好的办法是

enter image description here

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读