Condividi tramite


Tipi XML e ADO.NET nei contratti dati

Il modello di contratto dati Windows Communication Foundation (WCF) supporta alcuni tipi che rappresentano direttamente il formato XML. Quando questi tipi vengono serializzati in XML, il serializzatore scrive il contenuto XML di questi tipi senza ulteriore elaborazione. I tipi supportati sono XmlElement, matrici di XmlNode (ma non il tipo XmlNode stesso) e tipi che implementano IXmlSerializable. I tipi DataSet e DataTable, nonché i dataset tipizzati, vengono comunemente utilizzati nella programmazione dei database. Questi tipi implementano l'interfaccia IXmlSerializable e sono pertanto serializzabili nel modello del contratto dati. Alcune considerazioni speciali per questi tipi sono elencate alla fine di questo argomento.

Tipi XML

Elemento XML

Il tipo XmlElement viene serializzato utilizzando il contenuto XML. Si utilizzi ad esempio il tipo seguente.

<DataContract([Namespace] := "http://schemas.contoso.com")>  _
Public Class MyDataContract
    <DataMember()>  _
    Public myDataMember As XmlElement
    
    Public Sub TestClass() 
        Dim xd As New XmlDocument()
        myDataMember = xd.CreateElement("myElement")
        myDataMember.InnerText = "myContents"
        myDataMember.SetAttribute("myAttribute", "myValue")
    
    End Sub 
End Class 
[DataContract(Namespace=@"http://schemas.contoso.com")]
public class MyDataContract
{
    [DataMember]
    public XmlElement myDataMember;
    public void TestClass()
    {
        XmlDocument xd = new XmlDocument();
        myDataMember = xd.CreateElement("myElement");
        myDataMember.InnerText = "myContents";
        myDataMember.SetAttribute
         ("myAttribute","myValue");
    }
}

Questo tipo viene serializzato in XML come segue:

<MyDataContract xmlns="http://schemas.contoso.com">
    <myDataMember>
        <myElement  myAttribute="myValue">
            myContents
        </myElement>
    </myDataMember>
</MyDataContract>

Si noti che è ancora presente un elemento <myDataMember> membro dati del wrapper. Non è possibile rimuovere questo elemento dal modello del contratto dati. I serializzatori che gestiscono questo modello (DataContractSerializer e NetDataContractSerializer) possono generare attributi speciali in questo elemento wrapper. Tali attributi includono l'attributo "nil" standard dell'istanza di XML Schema (consentendo a XmlElement di essere null) e l'attributo "type" (consentendo l'uso di XmlElement in modo polimorfico). Inoltre, gli attributi XML seguenti sono specifici di WCF: "Id", "Ref", "Type" e "Assembly". Questi attributi possono essere generati per supportare l'utilizzo di XmlElement con la modalità di mantenimento dell'oggetto grafico abilitata o con NetDataContractSerializer. (Per ulteriori informazioni su sulla modalità di mantenimento dell'oggetto grafico, vedere Serializzazione e deserializzazione.)

Le matrici o raccolte di XmlElement sono consentite e sono gestite come qualsiasi altra matrice o raccolta. Ovvero, è presente un elemento wrapper per l'intera raccolta e un elemento wrapper separato (simile a <myDataMember>XmlElement nell'esempio precedente) per ogni della matrice.

Durante la deserializzazione, viene creato un XmlElement dal deserializzatore per il file XML in ingresso. Dal deserializzatore viene fornito un elemento XmlDocument padre valido.

Verificare che il frammento XML che viene deserializzato in un XmlElement definisca tutti i prefissi utilizzati e non si basi su definizioni di prefissi degli elementi predecessori. Questo aspetto è importante solo quando si adopera DataContractSerializer per utilizzare il file XML da un'origine diversa (non DataContractSerializer).

Se utilizzato con DataContractSerializer, XmlElement può essere assegnato in modo polimorfico, ma solo a un membro dati di tipo Object. Sebbene implementi IEnumerable, XmlElement non può essere utilizzato come tipo di raccolta e non può essere assegnato a un membro dati IEnumerable. In modo analogo alle assegnazioni polimorfiche, DataContractSerializer genera il nome del contratto dati nel file XML risultante, in questo caso è "XmlElement" nello spazio dei nomi "http://schemas.datacontract.org/2004/07/System.Xml".

Con NetDataContractSerializer, è supportata qualsiasi assegnazione polimorfica valida di XmlElement (Object o IEnumerable).

Non tentare di utilizzare uno dei serializzatori con i tipi derivati da XmlElement, indipendentemente dal fatto che siano stati assegnati in modo polimorfico o meno.

Matrice di XmlNode

L'utilizzo di matrici di XmlNode è molto simile all'utilizzo di XmlElement. Tuttavia, la scelta di utilizzare matrici di XmlNode garantisce una maggiore flessibilità rispetto all'utilizzo di XmlElement. È possibile scrivere più elementi nell'elemento wrapper del membro dati E’ possibile inserire al suo interno un contenuto diverso dagli elementi, ad esempio commenti XML. Infine, è inoltre possibile inserire attributi nell'elemento wrapper del membro dati. Tutte queste operazioni possono essere realizzate popolando la matrice di XmlNode con classi derivate specifiche di XmlNode, ad esempio XmlAttribute, XmlElement o XmlComment. Si utilizzi ad esempio il tipo seguente.

<DataContract([Namespace] := "http://schemas.contoso.com")>  _
Public Class MyDataContract
    <DataMember()>  _
    Public myDataMember(3) As XmlNode
    
    Public Sub TestClass() 
        Dim xd As New XmlDocument()
        Dim xe As XmlElement = xd.CreateElement("myElement")
        xe.InnerText = "myContents"
        xe.SetAttribute("myAttribute", "myValue")
        
        Dim atr As XmlAttribute = xe.Attributes(0)
        Dim cmnt As XmlComment = xd.CreateComment("myComment")
        
        myDataMember(0) = atr
        myDataMember(1) = cmnt
        myDataMember(2) = xe
        myDataMember(3) = xe
    
    End Sub 
    
End Class 
[DataContract(Namespace="http://schemas.contoso.com")]
public class MyDataContract
{
    [DataMember]
    public XmlNode[] myDataMember = new XmlNode[4];
    public void TestClass()
    {
        XmlDocument xd = new XmlDocument();
        XmlElement xe = xd.CreateElement("myElement");
        xe.InnerText = "myContents";
        xe.SetAttribute
         ("myAttribute","myValue");
      
        XmlAttribute atr = xe.Attributes[0];
        XmlComment cmnt = xd.CreateComment("myComment");
        
      myDataMember[0] = atr;
      myDataMember[1] = cmnt;
      myDataMember[2] = xe;
      myDataMember[3] = xe;
    }
}

Quando viene serializzato, l'XML risultante è simile al codice seguente.

<MyDataContract xmlns="http://schemas.contoso.com">
  <myDataMember myAttribute="myValue">
     <!--myComment-->
     <myElement  myAttribute="myValue">
 myContents
     </myElement>
     <myElement  myAttribute="myValue">
       myContents
     </myElement>
  </myDataMember>
</MyDataContract>

Si noti che l'elemento wrapper del membro dati <myDataMember> contiene un attributo, un commento e due elementi. Sono le quattro istanze di XmlNode serializzate.

Non è possibile serializzare una matrice di XmlNode che causa un XML non valido. Ad esempio, una matrice di due istanze di XmlNode di cui la prima è un XmlElement e la seconda è un XmlAttribute non è valida poiché questa sequenza non corrisponde ad alcuna istanza XML valida (non è possibile allegare un attributo).

Durante la deserializzazione di una matrice di XmlNode, i nodi vengono creati e popolati con informazioni derivate dal file XML in ingresso. Dal deserializzatore viene fornito un elemento XmlDocument padre valido. Tutti i nodi vengono deserializzati, includendo qualsiasi attributo sull'elemento wrapper del membro dati, ma escludendo gli attributi posizionati dai serializzatori WCF (ad esempio gli attributi utilizzati per indicare l'assegnazione polimorfica). La necessità di definire tutti i prefissi degli spazi dei nomi nel frammento XML si applica alla deserializzazione delle matrici di XmlNode e alla deserializzazione di XmlElement.

Se si utilizzano serializzatori per i quali è attivato il mantenimento dell'oggetto grafico, l'uguaglianza degli oggetti viene mantenuta solo a livello delle matrici di XmlNode e non per le singole istanze di XmlNode.

Non tentare di serializzare una matrice di XmlNode se uno o più nodi è impostato su null. L'intero membro matrice può essere null, ma non i singoli XmlNode contenuti nella matrice. Se l'intero membro matrice è Null, l'elemento wrapper del membro dati contiene un attributo speciale che indica che è Null. Durante la deserializzazione, anche l'intero membro matrice diviene Null.

Solo le matrici normali di XmlNode vengono trattate in modo speciale dal serializzatore. I membri dati dichiarati come altri tipi di raccolta contenenti XmlNode o i membri dati dichiarati come matrici di tipi derivati da XmlNode non vengono trattati in modo speciale. Pertanto, in genere non sono serializzabili a meno che soddisfino anche uno degli altri criteri di serializzazione.

Sono consentite matrici o raccolte di matrici di XmlNode. E’ presente un elemento wrapper per l'intera raccolta e un elemento wrapper separato (simile a <myDataMember>XmlNode nell'esempio precedente) per ogni matrice di nella matrice esterna o nella raccolta.

Se si popola un membro dati di tipo Array di Object o Array di IEnumerable con le istanze di XmlNode, il membro dati non viene trattato come una Array di istanze di XmlNode. Ogni membro della matrice viene serializzato separatamente.

Se si utilizzano con DataContractSerializer, le matrici di XmlNode possono essere assegnate in modo polimorfico, ma solo a un membro dati di tipo Object. Sebbene implementi IEnumerable, una matrice di XmlNode non può essere utilizzata come tipo di raccolta e non può essere assegnata a un membro dati IEnumerable. In modo analogo alle assegnazioni polimorfiche, DataContractSerializer genera il nome del contratto dati nel file XML risultante, in questo caso è "ArrayOfXmlNode" nello spazio dei nomi "http://schemas.datacontract.org/2004/07/System.Xml". Se utilizzata con la classe NetDataContractSerializer, qualsiasi assegnazione valida di una matrice di XmlNode è supportata.

Considerazioni sugli schemi

Per dettagli sul mapping dello schema dei tipi XML, vedere Riferimento allo schema del contratto dati. In questa sezione viene fornito un riepilogo dei punti importanti.

Un membro dati di tipo XmlElement viene mappato a un elemento definito utilizzando il tipo anonimo seguente.

<xsd:complexType>
   <xsd:sequence>
      <xsd:any minOccurs="0" processContents="lax" />
   </xsd:sequence>
</xsd:complexType>

Un membro dati di tipo matrice di XmlNode viene mappato a un elemento definito utilizzando il tipo anonimo seguente.

<xsd:complexType mixed="true">
   <xsd:sequence>
      <xsd:any minOccurs="0" maxOccurs="unbounded" processContents="lax" />
   </xsd:sequence>
   <xsd:anyAttribute/>
</xsd:complexType>

Tipi che implementano l'interfaccia IXmlSerializable

I tipi che implementano l'interfaccia IXmlSerializable sono completamente supportati da DataContractSerializer. L'attributo XmlSchemaProviderAttribute deve essere applicato sempre a questi tipi per controllarne lo schema.

Esistono tre varietà di tipi che implementano IXmlSerializable: i tipi che rappresentano contenuto arbitrario, i tipi che rappresentano un singolo elemento e i tipi DataSet legacy.

  • I tipi di contenuto utilizzano un metodo del provider dello schema specificato dall'attributo XmlSchemaProviderAttribute. Il metodo non restituisce null e la proprietà IsAny dell'attributo mantiene il valore predefinito false. Si tratta dell'utilizzo più comune di tipi IXmlSerializable.

  • I tipi di elemento vengono utilizzati quando un tipo IXmlSerializable deve controllare il nome del relativo elemento radice. Per contrassegnare un tipo come tipo di elemento, impostare la proprietà IsAny dell'attributo XmlSchemaProviderAttribute su true o fare in modo che il metodo del provider dello schema restituisca Null. Il metodo del provider dello schema è facoltativo per i tipi di elemento; è infatti possibile specificare Null anziché il nome del metodo in XmlSchemaProviderAttribute. Tuttavia, se IsAny è true ed è specificato un metodo del provider dello schema, il metodo deve restituire Null.

  • I tipi DataSet legacy sono tipi IXmlSerializable non contrassegnati con l'attributo XmlSchemaProviderAttribute, che si basano sul metodo GetSchema per la generazione dello schema. Questo modello è utilizzato per il tipo DataSet e le relative classi derivate del dataset tipizzato nelle versioni precedenti di .NET Framework, ma ora è obsoleto ed è supportato solo per elementi legacy. Non è consigliabile basarsi su questo modello, bensì applicare sempre XmlSchemaProviderAttribute ai tipi IXmlSerializable.

Tipi di contenuto IXmlSerializable

Quando si serializza un membro dati di un tipo che implementa IXmlSerializable e il tipo di contenuto rispecchia la definizione precedente, il serializzatore scrive l'elemento wrapper per il membro dati e passa il controllo al metodo WriteXml. L'implementazione WriteXml può scrivere qualsiasi elemento XML e può anche aggiungere attributi all'elemento wrapper. Quando l'operazione WriteXml è stata completata, il serializzatore chiude l'elemento.

Quando si deserializza un membro dati di un tipo che implementa IXmlSerializable e il tipo di contenuto rispecchia la definizione precedente, il deserializzatore posiziona il lettore XML sull'elemento wrapper per il membro dati e passa il controllo al metodo ReadXml. Il metodo deve leggere l'elemento intero, inclusi i tag di inizio e fine. Verificare che il codice ReadXml gestisca l'eventualità che l'elemento sia vuoto. Inoltre, l'implementazione ReadXml non deve basarsi sull'utilizzo di un nome specifico per l'elemento wrapper, poiché il nome viene scelto dal serializzatore e quindi può variare.

È consentito assegnare tipi di contenuto IXmlSerializable in modo polimorfico, ad esempio, ai membri dati di tipo Object. È inoltre consentito che le istanze del tipo siano Null. Infine, è possibile utilizzare tipi IXmlSerializable con il mantenimento dell'oggetto grafico attivato e con NetDataContractSerializer. Tutte queste funzionalità richiedono che il serializzatore WCF inserisca determinati attributi nell'elemento wrapper ("nil" e "type" nello spazio dei nomi dell'istanza dell'XML Schema e "Id", "Ref", "Type" e "Assembly" in uno spazio dei nomi specifico per WCF).

Attributi da ignorare quando si implementa ReadXml

Prima di passare il controllo al codice ReadXml, il deserializzatore esamina l'elemento XML, rileva questi attributi XML speciali e interviene su di essi. Ad esempio, se "nil" è true, un valore Null viene deserializzato e ReadXml non viene chiamato. Se viene rilevato il polimorfismo, il contenuto dell'elemento viene deserializzato come se si trattasse di un tipo diverso. L'implementazione di ReadXml del tipo assegnato in modo polimorfico viene chiamata. In ogni caso, un'implementazione ReadXml deve ignorare questi attributi speciali poiché vengono gestiti dal deserializzatore.

Considerazioni sullo schema per i tipi di contenuto IXmlSerializable

Se si esporta lo schema per un tipo di contenuto IXmlSerializable, viene chiamato il metodo del provider dello schema. Al metodo del provider dello schema viene passata una classe XmlSchemaSet. Il metodo può aggiungere qualsiasi schema valido al set di schemi. Il set di schemi contiene lo schema già conosciuto al momento dell'esportazione dello schema. Quando il metodo del provider dello schema deve aggiungere un elemento al set di schemi, deve determinare se esiste una classe XmlSchema con lo spazio dei nomi appropriato nel set. In tal caso, il metodo del provider dello schema deve aggiungere il nuovo elemento alla classe XmlSchema esistente. In caso contrario, deve creare una nuova istanza di XmlSchema. Questo è importante se vengono utilizzate matrici di tipi IXmlSerializable. Ad esempio, se un tipo IXmlSerializable viene esportato come tipo "A" nello spazio dei nomi "B", è possibile che quando il metodo del provider dello schema viene chiamato il set di schemi contenga già lo schema per "B" che contiene il tipo "ArrayOfA".

Oltre ad aggiungere i tipi nella classe XmlSchemaSet, il metodo del provider dello schema per i tipi di contenuto deve restituire un valore diverso da Null. Può restituire un oggetto XmlQualifiedName che specifica il nome del tipo di schema da utilizzare per il tipo IXmlSerializable specificato. Questo nome completo serve anche come nome e spazio dei nomi del contratto dati per il tipo. È consentito restituire un tipo che non esiste nel set di schemi quando il metodo del provider di schema viene restituito. Tuttavia, in genere al momento dell'esportazione di tutti i tipi correlati (il metodo Export viene chiamato per tutti i tipi attinenti su XsdDataContractExporter e si accede alla proprietà Schemas) il tipo esiste già nel set di schemi. L'accesso alla proprietà Schemas prima che tutte le chiamate Export attinenti siano state effettuate può generare un'eccezione XmlSchemaException. Per ulteriori informazioni su sul processo di esportazione, vedere Esportazione di schemi dalle classi.

Il metodo del provider dello schema può restituire anche l'oggetto XmlSchemaType da utilizzare. Il tipo può o meno essere anonimo. Se è anonimo, lo schema per il tipo IXmlSerializable viene esportato come tipo anonimo ogni volta che il tipo IXmlSerializable viene utilizzato come membro dati. Il tipo IXmlSerializable ha ancora un nome e uno spazio dei nomi del contratto dati (determinato come descritto in Nomi di contratto dati con l'eccezione che l'attributo DataContractAttribute non può essere utilizzato per personalizzare il nome). Se non è anonimo, deve essere uno dei tipi in XmlSchemaSet. Questo caso è equivalente alla restituzione del XmlQualifiedName del tipo.

Inoltre, viene esportata una dichiarazione di elemento globale per il tipo. Se al tipo non è applicato l'attributo XmlRootAttribute, l'elemento ha lo stesso nome e spazio dei nomi del contratto dati e la proprietà "nillable" sarà True. L'unica eccezione a questo comportamento è costituito dallo spazio dei nomi dello schema ("http://www.w3.org/2001/XMLSchema"), ovvero se il contratto dati del tipo è incluso in questo spazio dei nomi, l'elemento globale corrispondente si trova nello spazio dei nomi vuoto perché non è consentito aggiungere elementi nuovi allo spazio dei nomi dello schema. Se al tipo è applicato l'attributo XmlRootAttribute, la dichiarazione di elemento globale viene esportata utilizzando le proprietà ElementName, Namespace e IsNullable. L'impostazione predefinita quando è applicato l'attributo XmlRootAttribute è costituita dal nome del contratto dati, da un spazio dei nomi vuoto e da un valore "nillable" impostato su True.

Le stesse regole della dichiarazione di elemento globale si applicano ai tipi di dataset legacy. Si noti che XmlRootAttribute non può eseguire l'override delle dichiarazioni di elemento globale aggiunte tramite codice personalizzato o aggiunte a XmlSchemaSet utilizzando il metodo del provider dello schema o tramite GetSchema per i tipi di dataset legacy.

Tipi di elemento IXmlSerializable

I tipi di elemento IXmlSerializable hanno la proprietà IsAny impostata su true o il relativo metodo del provider dello schema restituisce null.

La serializzazione e deserializzazione di un tipo di elemento è molto simile alla serializzazione e deserializzazione di un tipo di contenuto. Esistono tuttavia alcune importanti differenze:

  • Si presuppone che l'implementazione WriteXml scriva un solo elemento (che può ovviamente contenere più elementi figlio). Non deve scrivere attributi all'esterno di questo singolo elemento, più elementi di pari livello o contenuto misto. L'elemento può essere vuoto.

  • L'implementazione ReadXml non deve leggere l'elemento wrapper, bensì deve leggere l'unico elemento prodotto da WriteXml.

  • Quando si serializza regolarmente un tipo di elemento (ad esempio, come un membro dati in un contratto dati), il serializzatore restituisce un elemento wrapper prima di chiamare WriteXml, come per i tipi di contenuto. Tuttavia, quando si serializza un tipo di elemento al primo livello, il serializzatore in genere non restituisce un elemento wrapper per l'elemento scritto da WriteXml, a meno che un nome e uno spazio dei nomi principali vengano specificati in modo esplicito durante la creazione del serializzatore nei costruttori DataContractSerializer o NetDataContractSerializer. Per ulteriori informazioni, vedere Serializzazione e deserializzazione.

  • Quando si serializza un tipo di elemento al primo livello senza specificare il nome e lo spazio dei nomi principali in fase di creazione, i metodi WriteStartObject e WriteEndObject essenzialmente non eseguono alcuna operazione e il metodo WriteObjectContent chiama WriteXml. In questa caso, l'oggetto serializzato non può essere Null e non può essere assegnato in modo polimorfico. Inoltre, il mantenimento dell'oggetto grafico non può essere abilitato e NetDataContractSerializer non può essere utilizzato.

  • Quando si deserializza un tipo di elemento al primo livello senza specificare il nome e lo spazio dei nomi principali in fase di creazione, il metodo IsStartObject restituisce true se riesce a trovare l'inizio di qualsiasi elemento. Il metodo ReadObject con il parametro verifyObjectName impostato su true si comporta allo stessa modo di IsStartObject prima di leggere effettivamente l'oggetto. ReadObject passa quindi il controllo al metodo ReadXml.

Lo schema esportato per i tipi di elemento è lo stesso del tipo XmlElement, come illustrato in una sezione precedente, con l'eccezione che il metodo del provider dello schema può aggiungere qualsiasi altro schema a XmlSchemaSet come con i tipi di contenuto. L'utilizzo dell'attributo XmlRootAttribute con i tipi di elemento non è consentito e le dichiarazioni di elemento globale non vengono mai generate per questi tipi.

Differenze rispetto a XmlSerializer

L'interfaccia IXmlSerializable e gli attributi XmlSchemaProviderAttribute e XmlRootAttribute vengono riconosciuti anche da XmlSerializer. Tuttavia, esistono alcune differenze nel modo in cui questi elementi vengono gestiti nel modello del contratto dati. Le differenze più importanti sono riepilogate di seguito:

  • Il metodo del provider dello schema deve essere pubblico per poter essere utilizzato in XmlSerializer, ma non deve essere pubblico per poter essere utilizzato nel modello del contratto dati.

  • Il metodo del provider dello schema viene chiamato quando IsAny è True nel modello del contratto dati, ma non con XmlSerializer.

  • Quando l'attributo XmlRootAttribute non è presente per contenuto o i tipi di dataset legacy, XmlSerializer esporta una dichiarazione di elemento globale nello spazio dei nomi vuoto. Nel modello del contratto dati, lo spazio dei nomi utilizzato è in genere lo spazio dei nomi del contratto dati, come descritto in precedenza.

Tenere presenti queste differenze durante la creazione di tipi che vengono utilizzati con entrambe le tecnologie di serializzazione.

Importazione dello schema IXmlSerializable

Quando si importa uno schema generato dai tipi IXmlSerializable, si verificano alcuni possibili scenari:

  • Lo schema generato può essere uno schema valido del contratto dati, come descritto in Riferimento allo schema del contratto dati. In questo caso, lo schema può essere importato normalmente e vengono generati tipi del contratto dati normali.

  • Lo schema generato può non essere uno schema valido del contratto dati. Ad esempio, il metodo del provider dello schema può generare lo schema coinvolgendo gli attributi XML che non sono supportati nel modello del contratto dati. In questo caso, è possibile importare lo schema come tipi IXmlSerializable. Questa modalità di importazione non è attiva per impostazione predefinita ma può essere attivata facilmente, ad esempio, con l'opzione della riga di comando /importXmlTypes in Strumento ServiceModel Metadata Utility Tool (Svcutil.exe). Questo aspetto viene descritto in dettaglio in Importazione dello schema per generare classi. Si noti che è necessario interagire direttamente con XML per le istanze del tipo. Si può anche scegliere di utilizzare una tecnologia di serializzazione diversa che supporti una gamma più ampia di schemi (vedere l'argomento relativo all'utilizzo di XmlSerializer).

  • Può essere necessario riutilizzare i tipi IXmlSerializable esistenti nel proxy anziché generare tipi nuovi. In questo caso, è possibile utilizzare la funzionalità dei tipi a cui è stato fatto riferimento descritta nell'argomento relativo all'importazione dello schema per generare i tipi per indicare il tipo da riutilizzare. Questa procedura corrisponde all'utilizzo dell'opzione /reference in svcutil.exe, che specifica l'assembly contenente i tipi da riutilizzare.

Rappresentazione di contenuto XML arbitrario nei contratti dati

XmlElement, la matrice di XmlNode e i tipi IXmlSerializable consentono di inserire contenuto XML arbitrario nel modello del contratto dati. DataContractSerializer e NetDataContractSerializer passano questo contenuto XML al writer XML in uso, senza interferire nel processo. Tuttavia, i writer XML possono applicare alcune restrizioni sul contenuto XML che scrivono. Di seguito sono riportati alcuni esempi importanti:

  • I writer XML in genere non consentono una dichiarazione del documento XML (ad esempio, <?xml version=’1.0’ ?>) nel mezzo della stesura di un altro documento. Non è possibile serializzare un documento XML completo come un elemento Array del membro dati XmlNode. Per questo scopo, è necessario rimuovere la dichiarazione del documento o utilizzare uno schema di codifica personalizzato per rappresentarlo.

  • Tutti i writer XML forniti con WCF rifiutano le istruzioni di elaborazione XML (<? … ?>) e definizioni del tipo di documento (<! … >), poiché non sono consentite nei messaggi SOAP. Anche in questo caso è possibile utilizzare un meccanismo di codifica personalizzato per evitare questa restrizione. Se è necessario includere questi elementi nel contenuto XML risultante, è possibile scrivere un codificatore personalizzato che utilizzi i writer XML che li supportano.

  • Quando si implementa WriteXml, evitare di chiamare il metodo WriteRaw sul writer XML. Poiché WCF utilizza una varietà di codifiche XML (inclusa la codifica binaria), è molto difficile o impossibile utilizzare WriteRaw in modo tale che il risultato sia utilizzabile in una codifica.

  • Quando si implementa WriteXml, evitare di utilizzare i metodi WriteEntityRef e WriteNmToken che sono non supportati dai writer XML forniti con WCF.

Utilizzo di dataset, dataset tipizzati e DataTable

L'utilizzo di questi tipi è supportato pienamente nel modello del contratto dati. Quando si utilizzano questi tipi, è opportuno considerare i punti seguenti:

  • Lo schema per questi tipi (in particolare DataSet e le relative classi derivate tipizzate) non può interagire con alcune piattaforme diverse da WCF o può determinare risultati insoddisfacenti se utilizzato con queste piattaforme. Inoltre, l'utilizzo del tipo DataSet può avere implicazioni sulle prestazioni. Può infine rendere più difficile il controllo della versione dell'applicazione in futuro. È consigliabile utilizzare tipi di contratto dati definiti in modo esplicito al posto dei tipi DataSet nei contratti.

  • Quando si importa lo schema DataSet o DataTable, è importante fare riferimento a questi tipi. Con l'utilità strumento da riga di comando svcutil.exe, è possibile eseguire questa operazione passando il nome dell'assembly System.Data.dll all'opzione /reference. Se si importa lo schema del dataset tipizzato, è necessario fare riferimento al tipo del dataset tipizzato. Con svcutil.exe, passare il percorso dell'assembly del dataset tipizzato all'opzione /reference. Per ulteriori informazioni su sui riferimenti ai tipi, vedere Importazione dello schema per generare classi.

Il supporto per i dataset tipizzati nel modello del contratto dati è limitato. I dataset tipizzati possono essere serializzati e deserializzati e possono esportare il proprio schema. Tuttavia, l’importazione dello schema del contratto dati non è in grado di generare nuovi tipi di dataset tipizzati dallo schema, in quanto può solo riutilizzare quelli esistenti. E’ possibile far riferimento a un dataset tipizzato esistente utilizzando l’opzione /r in Svcutil.exe. Se si tenta di utilizzare Svcutil.exe senza l’opzione /r in un servizio che utilizza un dataset tipizzato, viene automaticamente selezionato un serializzatore alternativo (XmlSerializer). Se è necessario utilizzare il DataContractSerializer e generare dataset dallo schema, è possible utilizzare la seguente procedura: generare i tipi di dataset tipizzati (utilizzando lo strumento Xsd.exe con l’opzione /d nel servizio), compilare i tipi, quindi farvi riferimento utilizzando l’opzione /r in Svcutil.exe.

Vedere anche

Riferimento

DataContractSerializer
IXmlSerializable

Concetti

Utilizzo di contratti dati
Tipi supportati dal serializzatore dei contratti dati