3

首先,我有两个项目,名称为我的解决方案的一部分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.csWindowsFormsApplication5我没有选择这个名字,如果听起来很乏味,请原谅我),我正在使用我提到的属性。没有命名空间问题,编译正常进行。

Form_Load然而,在of期间Form1,当我提到这些属性时,它们会抛出一个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])] });
     }
 }

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

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

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

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

从那时起,它就再也没有抛出这个编译器错误。我尝试了一切,包括使用ResXResourceWriter重新生成条目。就好像他们甚至没有被承认,而不是错误,我在运行时被告知字典属性总是空的!IRC 上的一位人士建议,该错误过去常常发生,因为ButtonBase不可序列化,即使Dictionary其本身是可序列化的。好吧,我从来没有要求编译器序列化字典。我想知道为什么会发生这种情况。

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

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上面提到的那些条目。

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

4

1 回答 1

1

这些属性是公共的还是派生自 或 的类FormUserControl成员Control

如果是这样,表单设计器可能会尝试将属性序列化到表单的 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>>();

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

于 2013-05-25T14:36:11.190 回答