Ho una struttura di oggetti che sto tentando di serializzare in xml che risulta in un livello di nodo duplicato. Sono abbastanza sicuro che abbia qualcosa a che fare con la sottoclasse perché ho dovuto implementare la mia deserializzazione ma non sono sicuro di cosa stia succedendo nella direzione opposta. La stessa struttura xml viene utilizzata come input quando deserializza i miei dati all'avvio dell'applicazione come quando la riserializza per essere salvata in seguito.La serializzazione XML genera nodi duplicati
Ecco ciò che il xml uscita difettosa assomiglia:
<Keyboarding xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<Exercises>
<Exercise>
<Exercise Id="3" ActivityNumber="5" SubActivityNumber="b" Type="Standard">
<Title>Test Title</Title>
<Instructions>Downloaded Update Instructions</Instructions>
</Exercise>
</Exercise>
</Exercises>
</Keyboarding>
Qui è quello che dovrebbe essere simile (e si presenta come il deserializzazione iniziale):
<Keyboarding xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<Exercises>
<Exercise Id="3" ActivityNumber="5" SubActivityNumber="b" Type="Standard">
<Title>Test Title</Title>
<Instructions>Downloaded Update Instructions</Instructions>
</Exercise>
</Exercises>
</Keyboarding>
L'oggetto principale contiene una raccolta di esercizi . L'esercizio è la mia classe base, mentre le sottoclassi sono determinate dall'attributo Tipo. Sto usando un serializzatore xml personalizzato per costruire gli oggetti di vari tipi derivati da Esercizio perché mi permette di usare il riflesso per abbinare qualsiasi delle dozzine circa di potenziali tipi derivati, che sono in ordine e quantità sconosciuti quando vengono letti per la prima volta dalla mia domanda.
Ecco la collezione del elemento root, utilizzando il serializzatore XML personalizzato:
[XmlArray("Exercises")]
[XmlArrayItem("Exercise", Type = typeof (ExerciseXmlSerializer<Exercise>))]
public Collection<Exercise> UnprocessedExercises { get; set; }
La mia classe di esercizio di base è dichiarato come segue:
[Serializable]
[XmlType(AnonymousType = true)]
public class Exercise
E la mia classe derivata è dichiarato come segue:
[Serializable]
[XmlType(AnonymousType = true)]
[XmlRoot(ElementName = "Exercise", IsNullable = false)]
public class StandardExercise : Exercise
Ecco la parte dello scrittore del mio serializzatore xml personalizzato:
public class ExerciseXmlSerializer<T> : IXmlSerializable where T : class
{
private T _data;
...
public void WriteXml(XmlWriter writer)
{
Type type = _data.GetType();
new XmlSerializer(type).Serialize(writer, _data);
}
...
}
Nel metodo WriteXml(), la variabile type è impostata correttamente sul tipo derivato, quindi perché crea un livello di nodo per il tipo di base e il tipo derivato? Come posso impedirgli di farlo?
La formattazione si interrompe quando si tenta di deserializzare i dati iniziali, che ha lo stesso formato xml. Quando ho modificato il codice e l'xml iniziale per abbinarlo, ha digitato tutto come tipo di base durante la deserializzazione e ha perso tutte le sottoclassi. Stavo specificando il tipo di elementi come serializzatore xml perché ci sono quasi 2 dozzine di possibili classi derivate a cui è possibile digitare e un numero sconosciuto di elementi e ordini nell'xml. Questo mi consente di prendere l'attributo Type e utilizzare reflection nel mio serializzatore personalizzato per creare gli oggetti derivati. Modificherò la mia domanda per cercare di chiarire un po '. – HotN
@HotN: hai provato a implementare il codice di serializzazione personalizzato direttamente nella classe Exercise? Ottengo eccezioni quando provo a eseguire il codice così com'è e questo è 'typeof (ExerciseXmlSerializer)' nell'XmlArrayItem che causa i problemi. Potresti anche provare a scrivere un xmlwriter personalizzato come mostrato nella risposta alla domanda collegata alla fine della mia risposta. –
Il problema è stato sicuramente causato dal 'typeof (ExerciseXmlSerializer)' parte, ma quello che ho trovato è che la modifica che a poco Esercizio risolverebbe il problema di serializzazione, ma avrebbe rotto il mio deserializzazione in quanto i tipi di classe derivati sono stati determinati da un attributo. L'ho fatto funzionare finalmente lasciando il codice originale in vigore per la collezione di esercizi ma implementando il mio XmlTextWriter, come suggerito. Sembra una specie di soluzione complicata, ma mi piace che non mi abbia richiesto di elencare 20 classi derivate come [XmlInclude]. Ad ogni modo, i risultati sono risultati. Grazie! –
HotN