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

时间:2013-05-25 14:18:15

标签: c# winforms dictionary resx

首先,作为我的解决方案的一部分,我有两个名为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的{​​{1}}中(我没有选择这个名字,请原谅我听起来很平淡),我正在使用我提到的属性。没有命名空间问题,正常情况下会进行编译。

WindowsFormsApplication5 Form_Load期间,当我引用这些属性时,他们会抛出Form1说 - :

NullReferenceException

第一次出现这些属性的方法是Object reference not set to an instance of an object. ,在表单加载时调用 -

WorkFlowPanel(string S, QuestionContainerState obj)

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

我之前曾与 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])] }); } } 中发生的Could not find a type for a name. The type was System.Collections.Dictionary...错误作斗争(我记不起整个错误)。显然,字典属性被序列化转换为二进制格式,并在resx文件中编码为base64,如下所示:

Form1.resx

我按照我在Stackoverflow上找到的解决方案,该解决方案告诉我将代码复制回旧版本。这是我从那以后一直在应用的修复。我发现如果我删除了这些条目(编译器错误要求我这样做),字典属性将始终为null。我觉得这与我目前的情况有关。碰巧在复制操作期间,resx文件条目以某种方式消失了。

从那以后,它从未抛出此编译器错误。我尝试了所有方法,包括使用<data name="MCQDisplay" mimetype="application/x-microsoft.net.object.binary.base64"> <value> AAEAAAD/////AQAAAAAAAAAEAQAA.... </value> <data name="MSQDisplay" .....> .... .... 重新生成条目。这就像他们甚至没有被承认,而不是错误,我告诉我在运行期间字典属性总是为空! IRC上的一个人建议错误发生,因为ResXResourceWriter不可序列化,即使ButtonBase本身就是{。}}。好吧,我从未要求编译器序列化字典。我想知道为什么会发生这种情况。

如果有人能帮助我解决这个该死的bug,我将非常感激。我脑子里也有这个模糊的解决方案,我不是将Dictionary个子项存储为值,而是可以存储类对象,它们具有镜像ButtonBase子项的text,optionId和checked属性的属性。但是,我认为这是最后的手段。

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

Form1.Designer.cs

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

EDIT_2 :我设法通过应用以下方法解决了这个问题 - :

  1. Matthew Watson的非序列化修复 - 如果编译器试图再次序列化字典属性。
  2. 我在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"))); 中注释了我上面提到的条目。
  3. 我接受Matthew的回答,因为我觉得非序列化修复是相关的。

1 个答案:

答案 0 :(得分:1)

这些属性是公开的还是来自FormUserControlControl的类的成员?

如果是这样,表单设计器可能会尝试将属性序列化为表单的Designer.cs文件。 (事实上​​,根据你的说法,它完全是可能的。)

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

[
    Browsable(false),
    DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)
]

public MyType MyPropertyWhichMustNotBeSerialized
{
    get;
    set;
}

你必须编辑它已经放在“.Designer.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>>();

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