【问题标题】:How is this field null?该字段如何为空?
【发布时间】:2016-03-24 05:46:58
【问题描述】:

我得到一个空异常,但该字段被初始化为一个空列表。那怎么可能是null呢?

错误出现在这个方法的第二行(在_hydradProperties上):

protected virtual void NotifyPropertyChanged<T>(Expression<Func<T>> expression)
{
    string propertyName = GetPropertyName(expression);

    if (!this._hydratedProperties.Contains(propertyName)) { this._hydratedProperties.Add(propertyName); }
}

这就是该字段的声明方式:

public abstract class EntityBase<TSubclass> : INotifyPropertyChanged where TSubclass : class
{
    private List<string> _hydratedProperties = new List<string>();

这是这样设置的:

    public Eta Eta
    {
        get { return this._eta; }

        set
        {
            this._eta = value;
            NotifyPropertyChanged(() => this.Eta);
        }
    }

这是完整的课程(删除了 cmets 和不相关的部分):

[DataContract]
public abstract class EntityBase<TSubclass> : INotifyPropertyChanged where TSubclass : class
{
    private List<string> _hydratedProperties = new List<string>();

    public bool IsPropertyHydrated(string propertyName)
    {
        return this._hydratedProperties.Contains(propertyName);
    }

    public event PropertyChangedEventHandler PropertyChanged;

    protected virtual void NotifyPropertyChanged<T>(Expression<Func<T>> expression)
    {
        string propertyName = GetPropertyName(expression);

        if (!this._hydratedProperties.Contains(propertyName)) { this._hydratedProperties.Add(propertyName); }

        PropertyChangedEventHandler handler = PropertyChanged;
        if (handler != null)
        {
            handler(this, new PropertyChangedEventArgs(propertyName));
        }
    }

    public string GetPropertyName<T>(Expression<Func<T>> expression)
    {
        MemberExpression memberExpression = (MemberExpression)expression.Body;
        return memberExpression.Member.Name;
    }
}

派生类:

[DataContract]
public class Bin : EntityBase<Bin>
{
    private Eta _eta;

    [DataMember]
    public Eta Eta
    {
        get { return this._eta; }

        set
        {
            this._eta = value;
            NotifyPropertyChanged(() => this.Eta);
        }
    }
}

【问题讨论】:

  • @BobHorn 您是否偶然从多个线程写入列表?如果是这样,您可能需要实现锁或使用线程安全集合
  • 我们确实需要准确的错误消息(包括提及的任何变量)和重现该错误的代表性代码示例。
  • 我想他知道什么是空异常,现在他知道空异常是_hidratedProperties。他需要知道为什么。从使用通知属性改变,可能是在数据绑定的情况。可能是线程问题,很难说。
  • @Grundy 这不是重复的。这个问题是问为什么 IT 抛出 NullReferenceException 而不是 NullReferenceException...

标签: c#


【解决方案1】:

我找到了一个比 Lucas 提供的更简单的答案。我不确定这个是否真的更好,但它很简单并且有效。我所做的只是将DataMember 属性添加到该字段。由于这指定该字段是数据协定的一部分,因此它包含在序列化/反序列化中,并且不再导致空引用错误。

[DataContract]
public abstract class EntityBase<TSubclass> : INotifyPropertyChanged where TSubclass : class
{
    [DataMember]
    private List<string> _hydratedProperties = new List<string>();

    // More code here
}

【讨论】:

  • 这也可以,但是属性列表显然也被序列化了。我不知道你的代码来判断这对你来说是好是坏,但它可能会产生你应该注意的意想不到的后果:如果你需要进行版本控制并且反序列化该类的先前版本,你会得到旧列表返回 - 您希望在该列表中以及您稍后添加的任何属性将不再存在 - 这可能会破坏您的代码,具体取决于您之后使用它执行的操作。它还将通过冗余数据增加有效负载大小。但是,如果所有这些对您来说都可以,那么当然,请继续。
  • 谢谢,卢卡斯。围绕此代码的逻辑仅在内存中,不会持久化,因此版本控制不是问题。不过,关于有效载荷,这是一个很好的观点。
【解决方案2】:

这是线索:

[DataContract]

是的。 DataContractSerializer does not call any constructor。相反,它使用FormatterServices.GetUninitializedObject 创建将被反序列化的对象。这绕过了构造函数调用。

你的初始化器:

private List<string> _hydratedProperties = new List<string>();

由编译器转换为隐式默认构造函数。

作为一种解决方法,您可以使用a deserialization callback with OnDeserializingAttribute

[DataContract]
public abstract class EntityBase<TSubclass> : INotifyPropertyChanged
    where TSubclass : class
{
    private List<string> _hydratedProperties;

    protected EntityBase()
    {
        Init();
    }

    private void Init()
    {
        _hydratedProperties = new List<string>()
    }

    [OnDeserializing]
    private void OnDeserializing(StreamingContext context)
    {
        Init();
    }

    // ... rest of code here
}

【讨论】:

  • 卢卡斯:我刚刚发布的答案呢?这也应该是一种可以接受的解决方法吗?
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2021-11-05
  • 2019-01-22
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多