Может ли кто-нибудь объяснить мне, почему этот первый пример будет сериализован в XML, а второй вызовет ошибки времени выполнения при попытке преобразовать все типы друг в друга? Если я удалю атрибуты XmlElement
из второго примера, он будет сериализован, но имя элемента XML будет неправильным («Элемент» вместо указанного для его типа). Первый фрагмент кода был создан с помощью инструмента XSD с использованием файла схемы.
А еще лучше, есть ли способ заставить это работать? Я бы предпочел использовать общие типы, приводящие к / от объектов. Это делает код намного чище. Явное приведение объектов показывает, что в вашем дизайне есть проблема.
public partial class OAIPMHtype
{
private object itemsField;
[XmlElement( "GetRecord", typeof( GetRecordType ) )]
[XmlElement( "Identify", typeof( IdentifyType ) )]
[XmlElement( "ListIdentifiers", typeof( ListIdentifiersType ) )]
[XmlElement( "ListMetadataFormats", typeof( ListMetadataFormatsType ) )]
[XmlElement( "ListRecords", typeof( ListRecordsType ) )]
[XmlElement( "ListSets", typeof( ListSetsType ) )]
[XmlElement( "error", typeof( OAIPMHerrorType ) )]
public object Item
{
get { return this.itemsField; }
set { this.itemsField = value; }
}
}
Это не будет сериализовано.
public class OaiPmh<T>
{
private T itemsField;
[XmlElement( "GetRecord", typeof( GetRecordType ) )]
[XmlElement( "Identify", typeof( IdentifyType ) )]
[XmlElement( "ListIdentifiers", typeof( ListIdentifiersType ) )]
[XmlElement( "ListMetadataFormats", typeof( ListMetadataFormatsType ) )]
[XmlElement( "ListRecords", typeof( ListRecordsType ) )]
[XmlElement( "ListSets", typeof( ListSetsType ) )]
[XmlElement( "error", typeof( OAIPMHerrorType ) )]
public T Item
{
get { return itemsField; }
set { itemsField = value; }
}
}
И для дальнейшего пояснения я попытался указать все дополнительные типы при создании объекта XmlSerializer
, и это не помогло.
Это вызванное исключение:
Unable to generate a temporary class (result=1).
error CS0030: Cannot convert type 'ErrorRequest' to 'GetRecordRequest'
error CS0030: Cannot convert type 'ErrorRequest' to 'ListRecordsRequest'
error CS0030: Cannot convert type 'ErrorRequest' to 'IdentityRequest'
error CS0030: Cannot convert type 'ErrorRequest' to 'ListSetsRequest'
error CS0030: Cannot convert type 'ErrorRequest' to 'ListIdentifiersRequest'
error CS0030: Cannot convert type 'ErrorRequest' to 'ListMetadataFormatsRequest'
error CS0029: Cannot implicitly convert type 'ListSetsRequest' to 'ErrorRequest'
error CS0029: Cannot implicitly convert type 'ListIdentifiersRequest' to 'ErrorRequest'
error CS0029: Cannot implicitly convert type 'ListMetadataFormatsRequest' to 'ErrorRequest'
error CS0029: Cannot implicitly convert type 'GetRecordRequest' to 'ErrorRequest'
error CS0029: Cannot implicitly convert type 'ListRecordsRequest' to 'ErrorRequest'
error CS0029: Cannot implicitly convert type 'IdentityRequest' to 'ErrorRequest
'
Это имеет смысл с универсальным типом, если посмотреть, как тип привязан к конкретному типу во время компиляции. Но, видя, как это работает со ссылкой на объект, на мой взгляд, он также должен работать с универсальным типом.