Información general sobre la arquitectura de transferencia de datos
Windows Communication Foundation (WCF) se puede considerar como una infraestructura de mensajería. Puede recibir mensajes, procesarlos y enviarlos al código de usuario para realizar acciones adicionales o puede construir mensajes a partir de los datos proporcionados por el código de usuario y entregarlos en un destino. Este tema, que está dirigido a desarrolladores avanzados, describe la arquitectura para el manejo de mensajes y los datos contenidos. Para obtener una vista más sencilla y orientada a tareas sobre cómo enviar y recibir datos, consulte Specifying Data Transfer in Service Contracts.
Nota
Este tema describe los detalles de implementación de WCF que no son visibles al examinar el modelo de objetos de WCF. Dos consejos se han de dar en relación a los detalles de implementación documentados. Primero, se simplifican las descripciones; la implementación real puede ser más compleja debido a optimizaciones u otras razones. Segundo, nunca debería basarse en detalles de implementación concretos, ni siquiera los documentados, porque puede que éstos cambien sin aviso de una versión a otra o incluso en un lanzamiento del servicio.
Arquitectura básica
En el núcleo de funcionalidades de control de mensajes de WCF está la clase Message, que se describe en detalle en Uso de la clase de mensajes. Los componentes en tiempo de ejecución de WCF pueden dividirse en dos partes principales: la pila de canales y el marco de trabajo del servicio, con la clase Message como punto de conexión.
La pila de canales es la responsable de la conversión entre una instancia Message válida y alguna acción que corresponda al envío o recepción de datos de mensajes. En el lado del envío, la pila de canales toma una instancia Message válida y, después de algo de procesado, realiza alguna acción que corresponde lógicamente con el envío del mensaje. La acción puede consistir en enviar paquetes TCP o HTTP, poner en cola el mensaje en Message Queuing, escribir el mensaje en una base de datos, guardarlo en un recurso compartido de archivos, o cualquier otra acción, dependiendo de la implementación. La acción más común es la de enviar el mensaje sobre un protocolo de red. En el lado de recepción, sucede lo contrario: se detecta una acción (que puede ser la llegada de paquetes TCP o HTTP o cualquier otra acción) y, tras el procesado, la pila de canales convierte esta acción en una instancia Message válida.
Puede utilizar WCF directamente con la pila de canales y la clase Message. Sin embargo, hacer esto es difícil y exige mucho tiempo. Además, el objeto Message no proporciona ninguna compatibilidad de metadatos, por lo que no puede generar clientes WCF fuertemente tipados si utiliza WCF de esta manera.
Por consiguiente, WCF incluye un marco de trabajo de servicio que proporciona un modelo de programación fácil de usar que puede utilizar para construir y recibir los objetos Message
. El marco de trabajo de servicio asigna servicios a los tipos .NET Framework a través de la noción de contratos de servicios y envía mensajes a las operaciones de usuarios que simplemente son métodos .NET Framework marcados con el atributo OperationContractAttribute. Para más detalles, consulte Diseño de contratos de servicios. Estos métodos pueden tener parámetros y valores devueltos. En el lado del servicio, el marco de trabajo de servicio convierte las instancias Message de entrada en parámetros y convierte los valores devueltos en instancias Message de salida. En el lado de cliente, hace lo contrario. Por ejemplo, considere la operación FindAirfare
descrita abajo.
[ServiceContract]
public interface IAirfareFinderService
{
[OperationContract]
int FindAirfare(string FromCity, string ToCity, out bool IsDirectFlight);
}
<ServiceContract()> _
Public Interface IAirfareFinderService
<OperationContract()> _
Function FindAirfare(ByVal FromCity As String, _
ByVal ToCity As String, ByRef IsDirectFlight As Boolean) As Integer
End Interface
Suponga que FindAirfare
se llama en el cliente. El marco de trabajo de servicio en el cliente convierte los parámetros FromCity
y ToCity
en una instancia de Message de salida y la pasa a la pila de canales que se va a enviar.
En el lado del servicio, cuando una instancia de Message llega de la pila de canales, el marco de trabajo de servicio extrae los datos pertinentes del mensaje para rellenar los parámetros FromCity
y ToCity
y, a continuación, llama al método FindAirfare
de lado de servicio. Cuando el método vuelve, el marco de trabajo de servicio toma el valor entero devuelto y el parámetro de salida IsDirectFlight
y crea una instancia de objeto de Message que contiene esta información. A continuación, pasa la instancia Message
a la pila de canales que se va a devolver al cliente.
En el lado de cliente, una instancia de Message que contiene el mensaje de respuesta emerge de la pila de canales. El marco de trabajo de servicio extrae el valor devuelto y el valor de IsDirectFlight
y los devuelve al llamador del cliente.
Message (clase)
La clase Message pretende ser una representación abstracta de un mensaje, pero su diseño está unido fuertemente al mensaje SOAP. Message contiene tres partes principales de información: un cuerpo del mensaje, encabezados del mensaje y propiedades de mensaje.
Cuerpo del mensaje
El objetivo del cuerpo del mensaje es representar la carga útil real del mensaje. El cuerpo del mensaje siempre se representa como un conjunto de información XML. Esto no significa que todos los mensajes creados o recibidos en WCF deban estar en formato XML. Depende de la pila de canales decidir cómo interpretar el cuerpo del mensaje. Puede emitirlo como XML, convertirlo a algún otro formato o incluso omitirlo completamente. Por supuesto, con la mayor parte de los enlaces que WCF proporciona, el cuerpo del mensaje se representa como contenido XML en la sección de cuerpo de una envoltura SOAP.
Es importante comprender que la clase Message
no necesariamente contiene un búfer con datos XML que representan el cuerpo. Lógicamente, Message
contiene un conjunto de información XML, pero este conjunto de información puede construirse dinámicamente y no existir nunca físicamente en memoria.
Colocar datos en el cuerpo del mensaje
No hay ningún mecanismo uniforme para colocar los datos en el cuerpo de un mensaje. La clase Message tiene un método abstracto, OnWriteBodyContents(XmlDictionaryWriter), que toma un XmlDictionaryWriter. Cada subclase de la clase Message es responsable de la invalidación de este método y de la escritura fuera de su propio contenido. El cuerpo del mensaje contiene lógicamente el conjunto de información XML que genera OnWriteBodyContent
. Por ejemplo, considere la siguiente subclase Message
.
public class AirfareRequestMessage : Message
{
public string fromCity = "Tokyo";
public string toCity = "London";
//code omitted…
protected override void OnWriteBodyContents(XmlDictionaryWriter w)
{
w.WriteStartElement("airfareRequest");
w.WriteElementString("from", fromCity);
w.WriteElementString("to", toCity);
w.WriteEndElement();
}
public override MessageVersion Version
{
get { throw new NotImplementedException("The method is not implemented.") ; }
}
public override MessageProperties Properties
{
get { throw new Exception("The method or operation is not implemented."); }
}
public override MessageHeaders Headers
{
get { throw new Exception("The method or operation is not implemented."); }
}
public override bool IsEmpty
{
get
{
return base.IsEmpty;
}
}
public override bool IsFault
{
get
{
return base.IsFault;
}
}
}
Public Class AirfareRequestMessage
Inherits Message
Public fromCity As String = "Tokyo"
Public toCity As String = "London"
' Code omitted…
Protected Overrides Sub OnWriteBodyContents(ByVal w As XmlDictionaryWriter)
w.WriteStartElement("airfareRequest")
w.WriteElementString("from", fromCity)
w.WriteElementString("to", toCity)
w.WriteEndElement()
End Sub
Public Overrides ReadOnly Property Version() As MessageVersion
Get
Throw New NotImplementedException("The method is not implemented.")
End Get
End Property
Public Overrides ReadOnly Property Properties() As MessageProperties
Get
Throw New Exception("The method or operation is not implemented.")
End Get
End Property
Public Overrides ReadOnly Property Headers() As MessageHeaders
Get
Throw New Exception("The method or operation is not implemented.")
End Get
End Property
Public Overrides ReadOnly Property IsEmpty() As Boolean
Get
Return MyBase.IsEmpty
End Get
End Property
Public Overrides ReadOnly Property IsFault() As Boolean
Get
Return MyBase.IsFault
End Get
End Property
End Class
Físicamente, una instancia AirfareRequestMessage
contiene solo dos cadenas ("fromCity" y "toCity"). Sin embargo, lógicamente el mensaje contiene el siguiente conjunto de información XML:
<airfareRequest>
<from>Tokyo</from>
<to>London</to>
</airfareRequest>
Por supuesto, no crearía normalmente mensajes de esta manera, porque puede utilizar el marco de trabajo del servicio para crear un mensaje como el anterior a partir de parámetros de contrato de operaciones. Además, la clase Message tiene métodos CreateMessage
estáticos que puede utilizar para crear mensajes con tipos comunes de contenido: un mensaje vacío, un mensaje que contiene un objeto serializado a XML con DataContractSerializer, un mensaje que contiene un error de SOAP, un mensaje que contiene XML representado por XmlReader, etc.
Obtención de datos del cuerpo de un mensaje
Puede extraer los datos almacenados en el cuerpo de un mensaje principalmente de dos maneras:
Puede obtener una vez el cuerpo del mensaje al completo llamando al método WriteBodyContents(XmlDictionaryWriter) y pasando un sistema de escritura de XML. El cuerpo del mensaje al completo se escribe en este sistema de escritura. Obtener el cuerpo del mensaje al completo de una vez también se llama escribir un mensaje. El canal de pilas realiza principalmente la escritura al enviar mensajes; una parte de la pila de canales tendrá, por lo general, acceso a todo el cuerpo del mensaje, lo codificará y enviará.
Otra manera de extraer información del cuerpo del mensaje es llamar GetReaderAtBodyContents() y obtener un lector XML. A continuación, se puede tener acceso al cuerpo del mensaje de manera consecutiva tanto como sea necesario llamando a los métodos en el lector. Obtener el cuerpo del mensaje parte por parte también se llama leer un mensaje. La lectura del mensaje es utilizada principalmente por el marco de trabajo de servicio al recibir mensajes. Por ejemplo, cuando DataContractSerializer se está utilizando, el marco de trabajo de servicio obtendrá un lector XML sobre el cuerpo y lo pasará al motor de deserialización, que comenzará, a continuación, a leer el mensaje elemento a elemento y a construir el gráfico de objeto correspondiente.
Solo se puede recuperar un cuerpo de mensaje una vez. Esto permite trabajar con secuencias de solo avance. Por ejemplo, puede escribir una invalidación OnWriteBodyContents(XmlDictionaryWriter) que lee de FileStream y devuelve los resultados como un conjunto de información XML. Nunca tendrá que "rebobinar" hasta el principio del archivo.
Los métodos WriteBodyContents
y GetReaderAtBodyContents
simplemente comprueban que el cuerpo del mensaje no se ha recuperado antes y, a continuación, llaman a OnWriteBodyContents
o a OnGetReaderAtBodyContents
, respectivamente.
Uso de mensajes en WCF
La mayoría de los mensajes pueden estar clasificados como salientes (aquéllos que crea el marco de trabajo de servicio para que la pila de canales los envíe) o entrantes (aquéllos que llegan desde pilas de canales y son interpretados por parte del marco de trabajo de servicio). Aún más, la pila de canales puede funcionar en modo almacenado en búfer o modo de transmisión por secuencias. El marco de trabajo de servicio también puede exponer un modelo de programación de transmisión por secuencias o sin transmisión por secuencias. Esto nos lleva a los casos enumerados en la siguiente tabla, junto con detalles simplificados de su implementación.
Tipo de mensaje | Datos del cuerpo en mensaje | Implementación de escritura (OnWriteBodyContents) | Implementación de lectura (OnGetReaderAtBodyContents) |
---|---|---|---|
Saliente, creado a partir del modelo de programación sin transmisión por secuencias | Los datos tienen que escribir el mensaje (por ejemplo, un objeto y la instancia de DataContractSerializer necesarios para serializarlo)* | Lógica personalizada para escribir el mensaje en función de los datos almacenados (por ejemplo, llamar WriteObject en DataContractSerializer si ése es el serializador en uso)* |
Llamar OnWriteBodyContents , almacenar en búfer los resultados, devolver un lector XML sobre el búfer |
Saliente, creado a partir del modelo de programación de transmisión por secuencias | La Stream con los datos que se han de escribir* |
Escriba los datos de la secuencia almacenada utilizando el mecanismo IStreamProvider * | Llamar OnWriteBodyContents , almacenar en búfer los resultados, devolver un lector XML sobre el búfer |
Entrante a partir de la pila de canales de transmisión por secuencias | Un objeto Stream que representa los datos que entran a través de la red con XmlReader sobre él |
Escriba fuera el contenido del XmlReader almacenado utilizando WriteNode |
Devuelve el XmlReader almacenado |
Entrante a partir de la pila de canales sin transmisión por secuencias | Un búfer que contiene datos del cuerpo con un XmlReader sobre él |
Escribe fuera el contenido del XmlReader almacenado utilizando WriteNode |
Devuelve el lenguaje almacenado |
*Estos elementos no se implementan directamente en subclases Message
, sino en subclases de la clase BodyWriter. Para obtener más información acerca de BodyWriter, vea Using the Message Class.
Encabezados de mensajes
Un mensaje puede contener encabezados. Un encabezado se compone lógicamente de un conjunto de información XML asociado a un nombre, un espacio de nombres y algunas propiedades. Se tiene acceso a los encabezados del mensaje utilizando la propiedad Headers
en Message. Cada encabezado está representado por una clase MessageHeader . Normalmente, los encabezados de mensajes están asignados a encabezados de mensaje SOAP al utilizar una pila de canales configurada para que funcione con mensajes SOAP.
Colocar información en un encabezado de mensaje y extraer información de él es similar a utilizar el cuerpo del mensaje. Se simplifica un poco el proceso porque no se admite la transmisión por secuencias. Es posible tener acceso más de una vez al contenido del mismo encabezado y se puede tener acceso a los encabezados en orden aleatorio, lo que hace que siempre estén almacenados en búfer. No hay ningún mecanismo de uso general disponible para obtener un lector XML sobre un encabezado, pero hay una subclase MessageHeader
interna a WCF que representa un encabezado legible con una función de este tipo. La pila del canal crea este tipo de MessageHeader
cuando llega un mensaje con encabezados personalizados de la aplicación. Esto permite al marco de trabajo del servicio usar un motor de deserialización, como DataContractSerializer, para interpretar estos encabezados.
Para más información, consulte Uso de la clase de mensajes.
Message (propiedades)
Un mensaje puede contener propiedades. Una propiedad es cualquier objeto de .NET Framework asociado a un nombre de cadena. Se tiene acceso a las propiedades a través de la propiedad Properties
en Message
.
A diferencia del cuerpo del mensaje y los encabezados del mensaje (que se asignan normalmente al cuerpo de SOAP y encabezados SOAP, respectivamente), las propiedades de mensajes normalmente no se envían ni reciben junto con los mensajes. Las propiedades de los mensajes existen principalmente como un mecanismo de comunicación para pasar datos sobre el mensaje entre los diversos canales de la pila de canales y entre la pila de canales y el modelo de servicio.
Por ejemplo, el canal del transporte HTTP incluido como parte de WCF es capaz de generar varios códigos de estado HTTP, como "404 (No encontrado)" y "500 (Error interno del servidor)", cuando envía respuestas a los clientes. Antes de enviar un mensaje de respuesta, comprueba para ver si las Properties
del Message
contienen una propiedad llamada a "httpResponse" que contiene un objeto de tipo HttpResponseMessageProperty. Si se encuentra este tipo de propiedad, mirará en la propiedad StatusCode y utilizará ese código de estado. Si no se encuentra, se utiliza el código "200 (Aceptar)" predeterminado.
Para más información, consulte Uso de la clase de mensajes.
El mensaje en conjunto
Hasta ahora, hemos discutido los métodos para tener acceso a las diversas partes del mensaje aislado. Sin embargo, la clase Message también proporciona métodos para trabajar con el mensaje completo como un todo. Por ejemplo, el método WriteMessage
escribe el mensaje completo en un sistema de escritura XML.
Para que esto sea posible, se debe definir una asignación entre la instancia Message
completa y un conjunto de información XML. Este tipo de asignación, de hecho, existe: WCF utiliza la norma de SOAP para definir esta asignación. Cuando una instancia Message
se escribe como un conjunto de información XML, el conjunto de información resultante es la envoltura SOAP válida que contiene el mensaje. Así, WriteMessage
realizaría normalmente los siguientes pasos:
Escribir la etiqueta de apertura del elemento de envoltura SOAP.
Escribir la etiqueta de apertura de elemento de encabezado SOAP, escribir todos los encabezados y cerrar el elemento de encabezado.
Escribir la etiqueta de apertura del elemento de cuerpo SOAP.
Llamar a
WriteBodyContents
o a un método equivalente para escribir el cuerpo.Cerrar los elementos de envoltura y cuerpo.
Los pasos anteriores están estrechamente unidos a la norma de SOAP. Esto es complicado, por el hecho de que existen múltiples versiones de SOAP, por ejemplo, es imposible escribir correctamente el elemento de envoltura de SOAP sin conocer la versión SOAP que se está utilizando. Asimismo, en algunos casos, puede ser deseable desactivar completamente esta compleja asignación específica del SOAP.
Para ello, se proporciona una propiedad Version
en Message
. Puede establecerse como la versión de SOAP que se va a utilizar cuando se escriba el mensaje, o puede establecerse como None
para evitar cualquier asignación específica de SOAP. Si la propiedad Version
se define como None
, los métodos que operan con todo el mensaje actúan como si el mensaje solo estuviera compuesto de su cuerpo, por ejemplo, WriteMessage
llamaría simplemente a WriteBodyContents
en lugar de realizar los diversos pasos enumerados anteriormente. Se espera que en los mensajes entrantes, Version
se detecte automáticamente y defina correctamente.
La pila de canales
Canales
Tal y como se dijo antes, la pila de canales es responsable de la conversión de instancias Message salientes en alguna acción (como enviar paquetes a través de la red) o de la conversión alguna acción (como recibir paquetes de la red) en instancias Message
entrantes.
La pila de canales consta de uno o más canales ordenados en una secuencia. Una instancia de Message
saliente se pasa al primer canal de la pila (también llamado canal más alto), que lo pasa al siguiente canal que esté por debajo en la pila, y así sucesivamente. El mensaje finaliza en el último canal, que se denomina el canal de transporte. Los mensajes entrantes se originan en el canal de transporte y se pasan de canal en canal hacia arriba en la pila. Desde el canal más alto, el mensaje se pasa normalmente al marco de trabajo de servicio. Aunque éste es el patrón usual para los mensajes de aplicaciones, algunos canales pueden trabajar de manera algo diferente, por ejemplo, pueden enviar sus propios mensajes de infraestructura sin tener que pasar un mensaje de un canal superior.
Los canales pueden funcionar en el mensaje de varias maneras a medida que atraviesa la pila. La operación más común es agregar un encabezado a un mensaje saliente y leer los encabezados en un mensaje entrante. Por ejemplo, un canal puede calcular la firma digital de un mensaje y agregarla como un encabezado. Un canal también puede inspeccionar este encabezado de firma digital en los mensajes entrantes y bloquear los mensajes que no tienen una firma para que no asciendan en la pila de canales. Los canales también definen o inspeccionan a menudo las propiedades de mensajes. Aunque está permitido, normalmente no se modifica el cuerpo de los mensajes, por ejemplo, el canal de seguridad de WCF puede cifrar el cuerpo del mensaje.
Canales de transporte y codificadores de mensajes
El canal más bajo de la pila es el responsable de la transformación real de un Messagesaliente, tal y como lo modificaron otros canales, en alguna acción. En el lado de recepción, éste es el canal que convierte alguna acción en un Message
que otros canales procesan.
Como se dijo anteriormente, se pueden variar las acciones: envío o recepción de paquetes de red a través de varios protocolos, lectura o escritura del mensaje en una base de datos o puesta o eliminación de la cola en una cola de Message Queuing, por mencionar unos pocos ejemplos. Todas estas acciones tienen algo en común: requieren una transformación entre la instancia de Message
de WCF y el verdadero grupo de bytes que se puede enviar, recibir, leer, escribir, poner en la cola o quitar de la cola. El proceso de conversión de un Message
en un grupo de bytes se denomina codificacióny el proceso inverso de crear Message
a partir de un grupo de bytes se denomina decodificación.
La mayoría de los canales de transporte usan componentes llamados codificadores de mensajes para realizar las tareas de codificación y decodificación. Un codificador de mensajes es una subclase de la clase MessageEncoder . MessageEncoder
incluye varias sobrecargas de método ReadMessage
y WriteMessage
para convertir entre Message
y grupos de bytes.
En el lado de envío, un canal de transporte de almacenado en búfer pasa el objeto Message
que recibió de un canal situado sobre él para WriteMessage
. Recibe una matriz de bytes, que utiliza a continuación para realizar su acción (como empaquetar estos bytes como paquetes TCP válidos y enviarlos al destino correcto). Un canal de transporte de transmisión por secuencia crea primero una Stream
(por ejemplo, sobre la conexión TCP de salida) y, a continuación, pasa la Stream
y el Message
que necesita para enviar a la sobrecarga WriteMessage
adecuada, que escribe el mensaje.
En el lado de recepción, un canal de transporte de almacenado en búfer extrae los bytes de entrada (por ejemplo, de los paquetes TCP entrantes) en una matriz y llama ReadMessage
para obtener un objeto Message
que puede pasar hacia arriba en la pila de canales. Un canal de transporte de transmisión por secuencias crea un objeto Stream
(por ejemplo, una secuencia de red sobre la conexión TCP entrante) y pasa eso a ReadMessage
para recibir un objeto Message
.
La separación entre los canales de transporte y el codificador de mensajes no es obligatoria; es posible escribir un canal de transporte que no use un codificador de mensajes. Sin embargo, la ventaja de esta separación es la facilidad de composición. Mientras un canal de transporte use solo el MessageEncoder, puede funcionar con cualquier codificador de WCF o de terceros. De igual modo, el mismo codificador puede usarse normalmente en cualquier canal de transporte.
Operación del codificador de mensajes
Para describir la operación típica de un codificador, es útil considerar los cuatro casos siguientes.
Operación | Comentario |
---|---|
Codificación, almacenado en búfer | En el modo de almacenado en búfer, el codificador crea normalmente un búfer de tamaño variable y, a continuación, crea un sistema de escritura de XML sobre él. A continuación, llama a WriteMessage(XmlWriter) en el mensaje que se está codificando que escribe los encabezados y, a continuación, el cuerpo mediante WriteBodyContents(XmlDictionaryWriter), como se explicó en la sección anterior sobre Message en este tema. El contenido del búfer (representado como una matriz de bytes) es devuelto a continuación para que lo use el canal de transporte. |
Codificación, secuenciada | En modo secuenciado, la operación es similar a la anterior, pero más sencilla. No hay necesidad de un búfer. Un sistema de escritura de XML se crea normalmente sobre la secuencia y se llama a WriteMessage(XmlWriter) en el Message para escribirlo en este sistema de escritura. |
Decodificación, almacenado en búfer | Al decodificar en modo de almacenado en búfer, se crea normalmente una subclase Message especial que contiene los datos almacenados en búfer. Se leen los encabezados del mensaje y se crea un lector XML colocado en el cuerpo del mensaje. Éste es el lector que se devolverá con GetReaderAtBodyContents(). |
Decodificación, secuenciada | Al decodificar en modo secuenciado, se crea normalmente una subclase Message especial. La secuencia se avanza lo suficiente para leer todos los encabezados y colocarlos en el cuerpo del mensaje. A continuación, se crea un lector XML sobre la secuencia. Éste es el lector que se devolverá con GetReaderAtBodyContents(). |
Los codificadores también pueden realizar otras funciones. Por ejemplo, los codificadores pueden agrupar sistemas de lectura y escritura XML. Es caro crear un nuevo sistema de lectura o escritura XML cada vez que se necesita. Por consiguiente, los codificadores mantienen normalmente un grupo de sistemas de lectura y escritura de tamaño configurable. En las descripciones de operación del codificador realizadas con anterioridad, cada vez que se usa la frase "cree un sistema de lectura/escritura XML", significa normalmente "tome uno del grupo o cree uno si no hay ninguno disponible". El codificador (y las subclases de Message
que crea al decodificar) contienen la lógica para devolver sistemas de lectura y escritura a los grupos una vez no se necesitan (por ejemplo, cuando se cierra el Message
).
WCF proporciona tres codificadores de mensajes, aunque es posible crear los tipos personalizados adicionales. Los tipos proporcionados son Texto, Binario y Mecanismo de optimización de transmisión de mensajes (MTOM). Estos se describen en detalle en Choosing a Message Encoder.
La interfaz IStreamProvider
Al escribir un mensaje saliente que contiene un cuerpo transmitido a un sistema de escritura XML, Message utilizará una secuencia de llamadas similar a la siguiente en su implementación OnWriteBodyContents(XmlDictionaryWriter) :
Escriba cualquier información necesaria que preceda a la secuencia (por ejemplo, la etiqueta de apertura XML)
Escriba la secuencia.
Escriba cualquier información que siga a la secuencia (por ejemplo, la etiqueta de cierre XML)
Esto funciona bien con codificaciones similares a la codificación XML textual. Sin embargo, hay algunas codificaciones que no proporcionan información del conjunto de información XML (por ejemplo, etiquetas para iniciar y finalizar elementos XML) junto con los datos contenidos en los elementos. Por ejemplo, para la codificación MTOM, la codificación del mensaje se divide en varias partes. Una parte contiene el conjunto de información XML, que puede contener referencias a otras partes de contenido de elementos reales. Puesto que el conjunto de información XML es normalmente pequeño en comparación con el contenido secuenciado, tiene sentido almacenar en búfer el conjunto de información, escribirlo y, a continuación, escribir el contenido de manera secuenciada. Esto significa que cuando se escribe la etiqueta del elemento de cierre, no se debería haber escrito todavía la secuencia.
Para este propósito, se utiliza la interfaz IStreamProvider . La interfaz tiene un método GetStream() que devuelve la secuencia que se va a escribir. La manera correcta de escribir un cuerpo de mensaje transmitido OnWriteBodyContents(XmlDictionaryWriter) es la siguiente:
Escriba cualquier información necesaria que preceda a la secuencia (por ejemplo, la etiqueta de apertura XML)
Llame a la sobrecarga
WriteValue
en el XmlDictionaryWriter que toma IStreamProvider, con una implementaciónIStreamProvider
que devuelve la secuencia que se va a escribir.Escriba cualquier información que siga a la secuencia (por ejemplo, la etiqueta de cierre XML)
Con este enfoque, el sistema de escritura XML puede escoger cuándo llamar GetStream() y escribir los datos transmitidos por secuencias. Por ejemplo, los sistemas de escritura XML textuales y binarios lo llamarán inmediatamente y escribirán el contenido transmitido por secuencias entre las etiquetas de inicio y cierre. El sistema de escritura de MTOM puede decidir llamar GetStream() posteriormente, cuando esté listo para escribir la parte adecuada del mensaje.
Representación de datos en el marco de trabajo de servicio
Como se mencionó en la sección "Arquitectura básica" de este tema, el marco de trabajo de servicio es la parte de WCF que, entre otras cosas, es responsable de la conversión entre un modelo de programación sencillo para datos de mensajes e instancias de Message
reales. Normalmente, un intercambio de mensajes se representa en el marco de trabajo de servicio como un método de .NET Framework marcado con el atributo OperationContractAttribute. El método puede tomar algunos parámetros y devolver un valor devuelto o parámetros out (o ambos). En el lado de servicio, los parámetros de entrada representan el mensaje entrante y el valor devuelto y los parámetros out representan el mensaje saliente. En el lado de cliente, se cumple lo contrario. El modelo de programación para la descripción de mensajes mediante el uso de parámetros y el valor devuelto se describe en detalle en Specifying Data Transfer in Service Contracts. Sin embargo, esta sección proporcionará una información general resumida.
Modelos de programación
El marco de servicio de WCF admite cinco modelos de programación diferentes para describir mensajes:
1. El mensaje vacío
Éste es el caso más simple. Para describir un mensaje entrante vacío, no utilice ningún parámetro de entrada.
[OperationContract]
int GetCurrentTemperature();
<OperationContract()> Function GetCurrentTemperature() As Integer
Para describir un mensaje saliente vacío, utilice un valor devuelto nulo y no utilice ningún parámetro out:
[OperationContract]
void SetDesiredTemperature(int t);
<OperationContract()> Sub SetDesiredTemperature(ByVal t As Integer)
Observe que esto es diferente de un contrato de operación unidireccional:
[OperationContract(IsOneWay = true)]
void SetLightbulb(bool isOn);
<OperationContract(IsOneWay:=True)> Sub SetLightbulb(ByVal isOn As Boolean)
En el ejemplo SetDesiredTemperature
, se describe un patrón de intercambio de mensajes bidireccional. Se devuelve un mensaje de la operación, pero está vacío. Es posible devolver un error de la operación. En el ejemplo "Definir bombilla", el patrón de intercambio de mensajes es unidireccional, por lo que no hay ningún mensaje saliente para describir. El servicio no puede devolver ningún estado al cliente en este caso.
2. Uso de la clase Message directamente
Es posible utilizar directamente la clase Message (o una de sus subclases) en un contrato de operación. En este caso, el marco de trabajo de servicio simplemente pasa el Message
de la operación a la pila de canales y viceversa, sin procesamiento adicional.
Hay dos casos principales para utilizar directamente Message
. Puede utilizar esto para escenarios avanzados, cuando ninguno de los otros modelos de programación le da suficiente flexibilidad para poder describir su mensaje. Por ejemplo, podría desear utilizar los archivos de un disco para describir un mensaje, con las propiedades de los archivos transformándose en encabezados del mensaje y el contenido de los archivos convirtiéndose en el cuerpo del mensaje. Puede crear a continuación algo similar a lo siguiente.
public class FileMessage : Message
// Code not shown.
Public Class FileMessage
Inherits Message
' Code not shown.
El segundo uso más extendido de un Message
en un contrato de operación tiene lugar cuando un servicio no se preocupa del contenido concreto del mensaje y actúa sobre el mensaje como sobre una caja negra. Por ejemplo, puede que tenga un servicio que reenvíe los mensajes a múltiples destinatarios. El contrato se puede escribir del siguiente modo.
[OperationContract]
public FileMessage GetFile()
{
//code omitted…
FileMessage fm = new FileMessage("myFile.xml");
return fm;
}
<OperationContract()> Public Function GetFile() As FileMessage
'code omitted…
Dim fm As New FileMessage("myFile.xml")
Return fm
End Function
En la práctica, la línea Action="*" desactiva el envío de mensajes y garantiza que todos los mensajes enviados al contrato IForwardingService
lleguen a su operación de ForwardMessage
. (Normalmente, el distribuidor examinaría el encabezado "Action" del mensaje para determinar para qué operación está pensado. Action="*" significa "todos los valores posibles del encabezado Action"). La combinación de Action="*" y el uso de Message como parámetro se conoce como el "contrato universal" porque es capaz de recibir todos los mensajes posibles. Para poder enviar todos los posibles mensajes, utilice Message como el valor devuelto y defina ReplyAction
como "*". Esto evitará que el marco de trabajo de servicio agregue su propio encabezado Action, lo que permite controlar este encabezado mediante el uso del objeto Message
devuelto.
3. Contratos de mensajes
WCF proporciona un modelo de programación declarativo para la descripción de mensajes, denominado contratos de mensajes. Este modelo se describe en detalle en Using Message Contracts. Esencialmente, el mensaje completo es representado por un único tipo de .NET Framework que utiliza atributos como MessageBodyMemberAttribute y MessageHeaderAttribute para describir qué partes de la clase del contrato de mensaje debería asignar a qué parte del mensaje.
Los contratos de mensajes proporcionan mucho control sobre las instancias Message
resultantes (aunque obviamente no tanto control como el resultante al utilizar directamente la clase Message
). Por ejemplo, los cuerpos de mensajes se crean a menudo a partir de varias partes de información, cada una representada por su propio elemento XML. Estos elementos pueden hallarse directamente en el cuerpo (modovacío ) o pueden ajustarse en un elemento XML que los abarque. Utilizar el modelo de programación de contrato de mensajes permite decidir entre desnudo y ajustado y controlar el nombre del nombre del contenedor y del espacio de nombres.
El siguiente ejemplo de código de un contrato de mensajes muestra estas características.
[MessageContract(IsWrapped = true, WrapperName = "Order")]
public class SubmitOrderMessage
{
[MessageHeader]
public string customerID;
[MessageBodyMember]
public string item;
[MessageBodyMember]
public int quantity;
}
<MessageContract(IsWrapped:=True, WrapperName:="Order")> _
Public Class SubmitOrderMessage
<MessageHeader()> Public customerID As String
<MessageBodyMember()> Public item As String
<MessageBodyMember()> Public quantity As Integer
End Class
Los elementos marcados para serializar (con MessageBodyMemberAttribute, MessageHeaderAttributeu otros atributos relacionados) deben ser serializables para participar en un contrato de mensaje. Para más información, consulte la sección "Serialización" más adelante en este tema.
4. Parámetros
A menudo, un desarrollador que desea describir una operación que actúa en varios pedazos de datos no necesita el grado de control que los contratos de mensajes proporcionan. Por ejemplo, al crear los nuevos servicios, uno normalmente no desea tomar la decisión de desnudo frente a ajustado, sino decidirse por el nombre del elemento contenedor. La toma de estas decisiones a menudo requiere un amplio conocimiento de servicios Web y SOAP.
El marco del servicio WCF puede escoger automáticamente la representación más adecuada e interoperable de SOAP para enviar o recibir varias piezas relacionadas de información, sin forzar al usuario a tomar estas decisiones. Para ello, basta con describir estos fragmentos de información como parámetros o valores devueltos de un contrato de operación. Por ejemplo, considere el siguiente contrato de operación:
[OperationContract]
void SubmitOrder(string customerID, string item, int quantity);
<OperationContract()> _
Sub SubmitOrder( _
ByVal customerID As String, _
ByVal item As String, _
ByVal quantity As Integer)
El marco de trabajo de servicio decide automáticamente colocar las tres piezas de información (customerID
, item
y quantity
) en el cuerpo del mensaje y ajustarlas en un elemento contenedor denominado SubmitOrderRequest
.
Describir la información que se va a enviar o recibir como una simple lista de parámetros de contratos de operaciones es el enfoque recomendado, a menos que tenga razones especiales para pasar al contrato de mensaje más complejo o a los modelos de programación basados en Message
.
5. Secuencia
Utilizar Stream
o una de sus subclases en un contrato de operación o como una sola parte del cuerpo del mensaje en un contrato de mensajes puede considerarse como un modelo de programación independiente de los descritos anteriormente. Utilizar Stream
de esta manera es la única manera de garantizar que su contrato se podrá usar de manera secuenciada, sin tener que escribir su propia subclase Message
compatible con transmisión por secuencias. Para más información, consulte Datos de gran tamaño y streaming.
Cuando Stream
o una de sus subclases se utiliza de esta manera, no se invoca el serializador. Para los mensajes salientes, se crea una subclase Message
de transmisión por secuencias y la secuencia se escribe tal y como se describe en la sección sobre la interfaz IStreamProvider . Para los mensajes entrantes, el marco de trabajo de servicio crea una subclase Stream
sobre el mensaje entrante y la proporciona a la operación.
Restricciones del modelo de programación
No se pueden combinar los modelos de programación descritos anteriormente de manera arbitraria. Por ejemplo, si una operación acepta un tipo de contrato de mensajes, el contrato de mensajes debe ser su único parámetro de entrada. Lo que es más, la operación debe devolver a continuación un mensaje vacío (tipo de valor devuelto de vacío) u otro contrato de mensajes. Estas restricciones del modelo de programación se describen en los temas de cada modelo de programación concreto: Using Message Contracts, Using the Message Classy Large Data and Streaming.
Formateadores de mensajes
Los modelos de programación descritos anteriormente se admiten agregando componentes llamados formateadores de mensajes en el marco de trabajo de servicio. Los formateadores de mensajes son tipos que implementan la interfaz IClientMessageFormatter o IDispatchMessageFormatter o ambas, para el uso en clientes y clientes WCF de servicio, respectivamente.
A los formateadores de mensajes se le agregan normalmente comportamientos. Por ejemplo, DataContractSerializerOperationBehavior agrega el formateador de mensajes de contratos de datos. Esto se hace en el lado de servicio estableciendo Formatter en el formateador correcto en el método ApplyDispatchBehavior(OperationDescription, DispatchOperation) o en el lado de cliente estableciendo Formatter en el formateador correcto en el método ApplyClientBehavior(OperationDescription, ClientOperation) .
Las siguientes tablas enumeran los métodos que un formateador de mensajes puede implementar.
Interfaz | Método | Acción |
---|---|---|
IDispatchMessageFormatter | DeserializeRequest(Message, Object[]) | Convierte un Message entrante en parámetros de operación |
IDispatchMessageFormatter | SerializeReply(MessageVersion, Object[], Object) | Crea un Message saliente a partir del valor devuelto/parámetros out de la operación |
IClientMessageFormatter | SerializeRequest(MessageVersion, Object[]) | Crea un Message saliente a partir de los parámetros de operación |
IClientMessageFormatter | DeserializeReply(Message, Object[]) | Convierte un Message entrante en un valor devuelto/parámetros out |
Serialización
Cuando use contratos de mensajes o parámetros para describir el contenido de mensajes, debe utilizar la serialización para convertir entre los tipos de .NET Framework y la representación del conjunto de información XML. La serialización se utiliza en otros lugares en WCF, por ejemplo, Message tiene un método GetBody genérico que puede utilizar para leer el cuerpo completo del mensaje deserializado en un objeto.
WCF admite dos tecnologías de serialización de fábrica para serializar y deserializar parámetros y partes de mensajes: DataContractSerializer y XmlSerializer
. Además, puede escribir serializadores personalizados. Sin embargo, otras partes de WCF (como el método GetBody
genérico o la serialización de errores de SOAP) pueden estar restringidas para utilizar solo las subclases XmlObjectSerializer (DataContractSerializer y NetDataContractSerializer, pero no XmlSerializer) o incluso puede estar codificado de manera no modificable para utilizar solo DataContractSerializer.
XmlSerializer
es el motor de serialización utilizado en los servicios Web de ASP.NET. DataContractSerializer
es el nuevo motor de serialización que entiende el nuevo modelo de programación de contrato de datos. DataContractSerializer
es la opción predeterminada, y la decisión de usar XmlSerializer
se puede tomar según la operación mediante el atributo DataContractFormatAttribute .
DataContractSerializerOperationBehavior y XmlSerializerOperationBehavior son los comportamientos de operaciones responsables de agregar los formateadores de mensajes para el DataContractSerializer
y XmlSerializer
, respectivamente. El comportamiento DataContractSerializerOperationBehavior puede realmente funcionar con cualquier serializador que derive de XmlObjectSerializer, incluso NetDataContractSerializer (descrito en detalle en Uso de serialización independiente). El comportamiento llama a una de las sobrecargas de método virtual CreateSerializer
para obtener el serializador. Para agregar un serializador diferente, cree una nueva subclase DataContractSerializerOperationBehavior e invalide ambas sobrecargas CreateSerializer
.