Я пытаюсь написать словарь общего назначения, который можно сохранить в файле app.config или web.config.Следуя примеру этого ответа , я написал словарь с некоторыми изменениями:
- Я использовал более современный
System.Xml.Linq
API - Я использовалHashtable, который позволяет мне хранить значение любого типа, при условии, что оно сериализуемо, а не только строки.
[SettingsSerializeAs(SettingsSerializeAs.Xml)]
public class XmlSerializableDictionary : Hashtable, IXmlSerializable
{
#region Constructors
/// <summary>
/// Initializes a new instance of <see cref="XmlSerializableDictionary"/>.
/// </summary>
public XmlSerializableDictionary()
{
}
#endregion
#region IXmlSerializable Members
/// <inheritdoc />
public XmlSchema GetSchema()
{
return null;
}
/// <inheritdoc />
public void ReadXml(XmlReader reader)
{
XDocument xdoc = XDocument.Load(reader);
foreach (XElement entry in xdoc.Elements())
{
XElement key = entry.Element(nameof(DictionaryEntry.Key))
?? throw new FormatException($"{nameof(DictionaryEntry.Key)} is missing in entry");
XElement valueElement = entry.Element(nameof(DictionaryEntry.Value))
?? throw new FormatException($"{nameof(DictionaryEntry.Value)} element is missing in entry");
XAttribute valueTypeName = valueElement?.Attribute("type") ?? throw new FormatException($"type attribute is missing in {nameof(DictionaryEntry.Value)} element");
if (valueTypeName.Value == "null")
{
this[key.Value] = null;
}
else
{
Type valueType = Type.GetType(valueTypeName.Value);
XmlSerializer xmlSerializer = new XmlSerializer(valueType ?? throw new ArgumentException("type attribute value is not a valid type name."));
Object value = xmlSerializer.Deserialize(valueElement.CreateReader());
this[key.Value] = value;
}
}
}
/// <inheritdoc />
public void WriteXml(XmlWriter writer)
{
XDocument xdoc = new XDocument();
xdoc.Add(new XElement(nameof(XmlSerializableDictionary)));
foreach (DictionaryEntry entry in this)
{
Type valueType = entry.Value?.GetType();
String typeName = valueType == null ? "null" : $"{valueType.FullName}, {valueType.Assembly.GetName().Name}";
Object value = null;
if (valueType != null)
{
XmlSerializer xmlSerializer = new XmlSerializer(valueType);
using (MemoryStream stream = new MemoryStream())
{
xmlSerializer.Serialize(stream, entry.Value);
using (StreamReader reader = new StreamReader(stream))
{
stream.Seek(0, SeekOrigin.Begin);
String xmlString = reader.ReadToEnd();
XElement valueXml = XElement.Load(xmlString);
value = valueXml;
}
}
}
XElement entryElement = new XElement(nameof(DictionaryEntry),
new XElement(nameof(DictionaryEntry.Key),
new XAttribute("type", typeName),
entry.Key.ToString()),
new XElement(nameof(DictionaryEntry.Value), value));
xdoc.Root?.Add(entryElement);
}
}
#endregion
/// <inheritdoc />
public override void Add(object key, object value)
{
if (!(key is String))
{
throw new NotSupportedException($"{nameof(key)} must be a string.");
}
base.Add(key, value);
}
}
Ожидаемая схема XML должна быть
<XmlSerializableDictionary>
<DictionaryEntry>
<Key>Key1</Key>
<Value type="System.Int32"> <!-- example type-->
Value1
</Value>
</DictionaryEntry>
<!--other entries here -->
</XmlSerializableDictionary>
Однако после добавления настройки этого типа в конструкторе настроек и примера значения этот параметр не записывается в файл app.config библиотеки классов, в которой это реализовано.Что-то мне не хватает?
Обновление Я обновил приведенный выше XML следующим образом, но он все равно не работает:
<?xml version="1.0" encoding="utf-16"?>
<XmlSerializableDictionary xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<DictionaryEntry>
<Key>Key1</Key>
<Value type="System.Int32">42</Value>
</DictionaryEntry>
</XmlSerializableDictionary>