尽管已实例化字典,字典属性始终为 null

本文关键字:字典 null 属性 实例化 | 更新日期: 2023-09-27 18:15:13

首先,我有两个名为ComponentsWindowsFormsApplication5的项目作为我解决方案的一部分(还有更多,但这些是无关紧要的(。我所指的字典在 Components 中,定义如下,以及它们的属性 -:

 Dictionary<int, ButtonBase> RespMCQ, DispMCQ; //<Question Number, Checkbox/Radiobutton>
 Dictionary<int, List<ButtonBase>> RespMSQ, DispMSQ;
 public Dictionary<int, ButtonBase> MCQResp
 {
     get
     {
         return RespMCQ;
     }
     set
     {
         RespMCQ = value;
     }
 }
 public Dictionary<int, List<ButtonBase>> MSQResponse
 {
     get
     {
         return RespMSQ;
     }
     set
     {
         RespMSQ = value;
     }
 }
 public Dictionary<int, ButtonBase> MCQDisplay
 {
     get
     {
         return DispMCQ;
     }
     set
     {
         DispMCQ = value;
     }
 }
 public Dictionary<int, List<ButtonBase>> MSQDisplay
 {
     get
     {
         return DispMSQ;
     }
     set
     {
         DispMSQ = value;
     }
 }

字典在 QuestionDisplay.cs 的构造函数中实例化,这是 Components 项目的一部分(这是它们所在的位置(。如下所示 -:

public QuestionDisplay()
{
    InitializeComponent();
    RespMCQ = new Dictionary<int, ButtonBase>();
    DispMCQ = new Dictionary<int, ButtonBase>();
    RespMSQ = new Dictionary<int, List<ButtonBase>>();
    DispMSQ = new Dictionary<int, List<ButtonBase>>();
    ....
}

Form1.cs WindowsFormsApplication5(我没有选择这个名字,如果听起来很平淡,请原谅我(,我正在使用我提到的属性。没有命名空间问题,编译正常进行。

然而,在Form1 Form_Load期间,当我提到这些属性时,他们抛出了一个NullReferenceException说法——:

Object reference not set to an instance of an object.

这些属性的第一次出现是在一个名为 WorkFlowPanel(string S, QuestionContainerState obj) 的方法中,该方法在表单加载时调用 - :

 if (QuesTypes[i].Equals("MCQ"))
 {
    string text = "";
    for (int ansCount = 0; ansCount < questions[i].Options.Count(); ansCount++)
    {
        if (Convert.ToInt32(K.Key).Equals(ansCount + 1))
        {
            text = questions[i].Options[ansCount];
            break;
        }
    }
    questionDisplay1.MCQResp.Add(i, new CustomRadio { optionId = Convert.ToInt32(K.Key), Checked = true, Text = text }); //Using anonymous class
 }
 else if (QuesTypes[i].Equals("MSQ") || QuesTypes[i].Equals("Linked"))
 {
     var storeoptions = K.Key.Split(':');
     questionDisplay1.MSQResponse.Add(i, new List<ButtonBase>());
     for (int s = 0; s < storeoptions.Count(); s++)
     {
         questionDisplay1.MSQResponse[i].Add(new CustomChecks { optionId = Convert.ToInt32(storeoptions[s]), Checked = true, Text = questions[i].Options[Convert.ToInt32(storeoptions[s])] });
     }
 }

令我惊讶的是,在调试时,我发现尽管字典被实例化,但属性始终为空。现在这个问题有一点历史,我将不得不深入研究,以提供背景。

我之前正在与Form1.resx中发生的Could not find a type for a name. The type was System.Collections.Dictionary...错误作斗争(我不记得整个错误(。显然,字典属性被序列化为二进制格式并在 resx 文件中编码为 base64,如下所示 -:

<data name="MCQDisplay" mimetype="application/x-microsoft.net.object.binary.base64">
<value>
        AAEAAAD/////AQAAAAAAAAAEAQAA....
</value>
<data name="MSQDisplay" .....>
....
....

我遵循了我在Stackoverflow上找到的解决方案,它告诉我将代码复制回旧版本。这是我从那时起一直在应用的修复程序。我发现如果我删除这些条目(编译器错误要求我这样做(,字典属性将始终为空。我觉得这与我目前的情况有关。碰巧的是,在复制操作期间,resx 文件条目不知何故消失了。

从那以后,它从未抛出过这个编译器错误。我尝试了所有方法,包括使用 ResXResourceWriter 来重新生成条目。就像它们甚至没有被确认,而不是错误,我在运行时被告知字典属性始终为 null!IRC上的一个人建议,过去发生错误是因为ButtonBase不可序列化,即使Dictionary本身是不可序列化的。好吧,我从来没有要求编译器序列化字典。我想知道为什么会发生这种情况。

如果有人能帮助我解决这个的错误,我将不胜感激。我脑海中还有一个模糊的解决方案,即我可以存储类对象,而不是将ButtonBase子对象存储为值,这些对象具有镜像ButtonBase子项的文本、optionId 和检查属性的属性。但是,我认为这是最后的手段。

EDIT_1:我不知道这是否相关,但我在Form1.Designer.cs中找到了这些条目 - :

this.questionDisplay1.MCQDisplay = ((System.Collections.Generic.Dictionary<int, System.Windows.Forms.ButtonBase>)(resources.GetObject("questionDisplay1.MCQDisplay")));
this.questionDisplay1.MCQResp = ((System.Collections.Generic.Dictionary<int, System.Windows.Forms.ButtonBase>)(resources.GetObject("questionDisplay1.MCQResp")));
this.questionDisplay1.MSQDisplay = ((System.Collections.Generic.Dictionary<int, System.Collections.Generic.List<System.Windows.Forms.ButtonBase>>)(resources.GetObject("questionDisplay1.MSQDisplay")));
this.questionDisplay1.MSQResponse = ((System.Collections.Generic.Dictionary<int, System.Collections.Generic.List<System.Windows.Forms.ButtonBase>>)(resources.GetObject("questionDisplay1.MSQResponse")));

我是否必须删除它们或以某种方式修改它们?我觉得这些与马修的回答有某种关系。

EDIT_2:我设法通过应用以下内容来解决此问题 - :

  1. Matthew Watson 的非序列化修复 - 以防编译器尝试再次序列化字典属性。
  2. 我注释掉了Form1.Designer.cs中的条目,我上面提到的条目。

我接受马修的回答,因为我觉得非序列化修复是相关的。

尽管已实例化字典,字典属性始终为 null

这些属性既是公共的,又是从FormUserControlControl派生的类的成员吗?

如果是这样,窗体设计器可能会尝试将属性序列化为窗体的设计器.cs文件。(事实上,从你所说的来看,这是完全可能的

要阻止这种情况发生,您可以使用DesignerSerializationVisibilityAttribute来防止它,例如:

[
    Browsable(false),
    DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)
]
public MyType MyPropertyWhichMustNotBeSerialized
{
    get;
    set;
}

您必须编辑掉它已经放在".设计器.cs"文件 - 添加此属性可能不会使其自动消失。

如果您没有定期检查是否需要添加此属性,我认为您可能希望检查直接或间接派生自Control的任何类中的所有公共属性,并在必要时添加此属性。


对于这个问题,空引用;是否有可能调用FormLoad()作为调用InitialiseComponent()的副作用?这将导致空引用错误。

为了最安全起见,您应该为属性使用支持字段,而不是自动属性。

然后,不要初始化构造函数中的字段,而是在声明字段的位置初始化它们,如下所示:

private Dictionary<int, ButtonBase>       respMCQ = new Dictionary<int, ButtonBase>();
private Dictionary<int, ButtonBase>       dispMCQ = new Dictionary<int, ButtonBase>();
private Dictionary<int, List<ButtonBase>> respMSQ = new Dictionary<int, List<ButtonBase>>();
private Dictionary<int, List<ButtonBase>> dispMSQ = new Dictionary<int, List<ButtonBase>>();

这样,可以保证在调用任何构造函数之前初始化它们。