Yan Cui
I help clients go faster for less using serverless technologies.
This article is brought to you by
Don’t reinvent the patterns. Catalyst gives you consistent APIs for messaging, data, and workflow with key microservice patterns like circuit-breakers and retries for free.
If you try to serialize/deserialize a type which uses the generic Dictionary<TKey, TValue> type with the XmlSerializer then you’ll get an InvalidOperationException, for instance:
Here’s my class:
public class MyClass { // need a parameterless constructor for serialization public MyClass() { MyDictionary = new Dictionary<string, string>(); } public Dictionary<string, string> MyDictionary { get; set; } }
I’ll get an InvalidOperationException with an inner exception of type NotSupportedException when I do this:
var xmlSerializer = new XmlSerializer(typeof(MyClass));
And the error message of the NotSupportedException is:
“Cannot serialize member MyClass.MyDictionary of type System.Collections.Generic.Dictionary`2[[System.String, mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089],[System.String, mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]], because it implements IDictionary.”
The reason why Dictionary<TKey, TValue> is not supported by the XmlSerializer is that types such as Dictionary, HashTable, etc. needs an equality comparer which can’t be serialized into XML easily and won’t be portable anyhow. To get around this problem, you generally have two options, in the order of my personal preference:
Use DataContractSerizalizer
The easiest, and cleanest solution is to switch over to data contract serialization, and the only change required would be to mark your class with the DataContractAttribute and mark the properties you want to serialize with the DataMemberAttribute. My class would now look like this:
[DataContract] public class MyClass { // need a parameterless constructor for serialization public MyClass() { MyDictionary = new Dictionary<string, string>(); } [DataMember] public Dictionary<string, string> MyDictionary { get; set; } }
And to serialize it into Xml, just use the DataContractSerializer and write the output with a XmlTextWriter like this:
var serializer = new DataContractSerializer(typeof(MyClass)); string xmlString; using (var sw = new StringWriter()) { using (var writer = new XmlTextWriter(sw)) { writer.Formatting = Formatting.Indented; // indent the Xml so it's human readable serializer.WriteObject(writer, marketplace); writer.Flush(); xmlString = sw.ToString(); } }
Use a custom Dictionary type
The alternative is to create your own Dictionary implementation which is Xml serializable. See the references section for an implementation Paul Welter has created.
References:
XML Serializable Generic Dictionary
Whenever you’re ready, here are 3 ways I can help you:
- Production-Ready Serverless: Join 20+ AWS Heroes & Community Builders and 1000+ other students in levelling up your serverless game. This is your one-stop shop for quickly levelling up your serverless skills.
- I help clients launch product ideas, improve their development processes and upskill their teams. If you’d like to work together, then let’s get in touch.
- Join my community on Discord, ask questions, and join the discussion on all things AWS and Serverless.
great stuff! exactly what i needed. thx for your work and posting.
serializer.WriteObject(writer, marketplace);
where does marketplace come from?
@Nanek – imagine ‘marketplace’ being an instance of ‘MyClass’, looks like I missed out a line or two of the code snippet
Thank you for sharing this.
Thanks for the information.
Just thought I’d add – you need to add a reference to System.Runtime.Serialization for this, and it is only available in .NET 3 upwards.
Pingback: [C#] Dictionary ?? Xml « Develope note
Thank you very much for this post. I really needed something like this.
Thank you. This helped me serialize an object not serializable with standard xml serializer.
Great!
I’d add that for DataContractSerizalizer a reference to System.Runtime.Serialization is necessary.
Thnaks, guy!
Thanks. how do you then read/deserialize it?
In .Net 4.5.1 I am getting
{“Type ‘System.Collections.Generic.Dictionary`2[[System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089],[System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]]’ with data contract name ‘ArrayOfKeyValueOfstringstring:http://schemas.microsoft.com/2003/10/Serialization/Arrays‘ is not expected. Consider using a DataContractResolver or add any types not known statically to the list of known types – for example, by using the KnownTypeAttribute attribute or by adding them to the list of known types passed to DataContractSerializer.”}
Here is the class I am trying to serialize
[DataContract]
public class KeywordFunctionMap
{
[DataMember]
public Dictionary Map { get; set; }
// need a parameterless constructor for serialization
public KeywordFunctionMap()
{
Map = new Dictionary();
}
}
I just put together a Console app in .Net 4.5.1 and it worked as expected, see https://gist.github.com/theburningmonk/3b1f1a33ba616bda7474
Pingback: Serialising complex types in C#
Thanks! How do I make desrialize tto Dicitonary?
Pingback: ?????????? – CodingBlog