尽管已实例化字典,字典属性始终为 null
本文关键字:字典 null 属性 实例化 | 更新日期: 2023-09-27 18:15:13
首先,我有两个名为Components
和WindowsFormsApplication5
的项目作为我解决方案的一部分(还有更多,但这些是无关紧要的(。我所指的字典在 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:我设法通过应用以下内容来解决此问题 - :
- Matthew Watson 的非序列化修复 - 以防编译器尝试再次序列化字典属性。
- 我注释掉了
Form1.Designer.cs
中的条目,我上面提到的条目。
我接受马修的回答,因为我觉得非序列化修复是相关的。
这些属性既是公共的,又是从Form
、UserControl
或Control
派生的类的成员吗?
如果是这样,窗体设计器可能会尝试将属性序列化为窗体的设计器.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>>();
这样,可以保证在调用任何构造函数之前初始化它们。