Как добавить атрибут XmlInclude динамически
У меня есть следующие классы
[XmlRoot]
public class AList
{
public List<B> ListOfBs {get; set;}
}
public class B
{
public string BaseProperty {get; set;}
}
public class C : B
{
public string SomeProperty {get; set;}
}
public class Main
{
public static void Main(string[] args)
{
var aList = new AList();
aList.ListOfBs = new List<B>();
var c = new C { BaseProperty = "Base", SomeProperty = "Some" };
aList.ListOfBs.Add(c);
var type = typeof (AList);
var serializer = new XmlSerializer(type);
TextWriter w = new StringWriter();
serializer.Serialize(w, aList);
}
}
Теперь, когда я пытаюсь запустить код, на последней строке я получил InvalidOperationException, в котором говорится, что
Тип XmlTest.C не ожидался. Используйте атрибут XmlInclude или SoapInclude, чтобы указать типы, которые не известны статически.
Я знаю, что добавление атрибута [XmlInclude (typeof (C))] с помощью [XmlRoot] решит проблему. Но я хочу добиться этого динамично. Потому что в моем проекте класс C не известен до загрузки. Класс C загружается как плагин, поэтому мне не удается добавить туда атрибут XmlInclude.
Я попробовал также с
TypeDescriptor.AddAttributes(typeof(AList), new[] { new XmlIncludeAttribute(c.GetType()) });
до
var type = typeof (AList);
но не используется. Он по-прежнему дает то же исключение.
Есть ли у кого-нибудь идеи о том, как его достичь?
Ответы
Ответ 1
Два варианта; самый простой (но нечетный xml):
XmlSerializer ser = new XmlSerializer(typeof(AList),
new Type[] {typeof(B), typeof(C)});
Пример вывода:
<AList xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<ListOfBs>
<B />
<B xsi:type="C" />
</ListOfBs>
</AList>
Более элегантным является:
XmlAttributeOverrides aor = new XmlAttributeOverrides();
XmlAttributes listAttribs = new XmlAttributes();
listAttribs.XmlElements.Add(new XmlElementAttribute("b", typeof(B)));
listAttribs.XmlElements.Add(new XmlElementAttribute("c", typeof(C)));
aor.Add(typeof(AList), "ListOfBs", listAttribs);
XmlSerializer ser = new XmlSerializer(typeof(AList), aor);
Пример вывода:
<AList xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<b />
<c />
</AList>
В любом случае вы должны кэшировать и повторно использовать экземпляр ser
; иначе вы будете геморрагировать память из динамической компиляции.
Ответ 2
Основываясь на первом ответе Марка (мне нужно только прочитать, поэтому мне не нужно предотвращать странный вывод), я использую более динамичный/общий тип-массив для учета неизвестных типов, вдохновленных этим codeproject.
public static XmlSerializer GetSerializer()
{
var lListOfBs = (from lAssembly in AppDomain.CurrentDomain.GetAssemblies()
from lType in lAssembly.GetTypes()
where typeof(B).IsAssignableFrom(lType)
select lType).ToArray();
return new XmlSerializer(typeof(AList), lListOfBs);
}
(Вероятно, можно сделать его более эффективным, например, используя статический или только для чтения тип-массив вместо локальной переменной. Это позволит избежать повторного использования Reflection. Но я не знаю достаточно, когда сборка загружается и классы и свойства инициализируются, чтобы знать, не приведет ли вас к неприятностям. Мое использование не так уж и много, чтобы потратить время на исследование всего этого, поэтому я просто использую одно и то же отражение несколько раз.)
Ответ 3
Посмотрите документацию XmlSerializer. Существует конструктор, который ожидает, что известные типы будут вторым параметром. Это должно работать нормально, если вы используете случай.
Ответ 4
Я не думаю, что атрибуты могут применяться во время выполнения, поскольку они используются для создания метаданных по коду CIL.