N niveles de LINQ to SQL con servicios Web
LINQ to SQL está diseñado especialmente para su uso en el nivel intermedio de una capa de acceso a datos (DAL) débilmente acoplada (por ejemplo, un servicio Web). Si el nivel de presentación es una página web ASP.NET, el control de servidor web LinqDataSource se utiliza para controlar la transferencia de datos entre la interfaz de usuario y LINQ to SQL en el de nivel intermedio. Si el nivel de presentación no es una página ASP.NET, entonces el nivel intermedio y el nivel de presentación deben realizar un trabajo adicional para administrar la serialización y deserialización de datos.
Establecer LINQ to SQL en el nivel intermedio
En un servicio Web o aplicación de n niveles, el nivel intermedio contiene el contexto de los datos y las clases de entidad. Puede crear estas clases manualmente o mediante SQLMetal.exe o el Object Relational Designer, como se describe en otra parte de la documentación. En tiempo de diseño, tiene la opción de hacer las clases de entidad serializables. Para obtener más información, vea Cómo: Serializar entidades. Otra opción consiste en crear un conjunto independiente de clases que encapsulan los datos que se van a serializar y, a continuación, proyectar en esos tipos serializables cuando se devuelvan datos en las consultas de LINQ.
A continuación, se define la interfaz con los métodos a los que los clientes llamarán para recuperar, insertar y actualizar datos. Los métodos de interfaz envuelven las consultas de LINQ. Puede utilizar cualquier tipo de mecanismo de serialización para administrar las llamadas a métodos remotos y la serialización de datos. El único requisito es que, si tiene relaciones cíclicas o bidireccionales en su modelo de objetos, como las que existen entre Clientes y Pedidos en el modelo de objetos estándar de Northwind, deberá utilizar un serializador que admita esa situación. Windows Communication Foundation (WCF) DataContractSerializer admite relaciones bidireccionales, pero no así el serializador XmlSerializer que se utiliza con servicios Web que no son de WCF. Si selecciona el serializador XmlSerializer, deberá asegurarse de que su modelo de objetos no contenga relaciones cíclicas.
Para obtener más información sobre Windows Communication Foundation, vea Windows Communication Foundation Services y servicios de datos de WCF en Visual Studio.
Implemente sus reglas de empresa u otra lógica específica del dominio utilizando las clases parciales y métodos de DataContext y las clases de entidad para conectar con eventos de tiempo de ejecución de LINQ to SQL. Para obtener más información, consulte Implementación de lógica de negocios de N niveles.
Definir los tipos serializables
El nivel de cliente o de presentación debe contar con las definiciones de tipo para las clases que recibirá desde el nivel intermedio. Esos tipos pueden ser clases de entidad propiamente dichas o clases especiales que envuelven sólo ciertos campos de las clases de entidad para la comunicación remota. En cualquier caso, a LINQ to SQL le resulta completamente indiferente cómo el nivel de presentación adquiere esas definiciones de tipo. Por ejemplo, el nivel de presentación podría utilizar WCF para generar los tipos automáticamente, o podría tener una copia de una DLL en la que se definen esos tipos, o simplemente podría definir sus propias versiones de los tipos.
Recuperar e insertar datos
El nivel intermedio define una interfaz que especifica cómo el nivel de presentación obtiene acceso a los datos. Por ejemplo, GetProductByID(int productID)
o GetCustomers()
. En el nivel intermedio, el cuerpo del método crea generalmente una nueva instancia de DataContext y ejecuta una consulta contra una o más de su tabla. A continuación, el nivel intermedio devuelve el resultado como un IEnumerable<T>, donde T
es una clase de entidad u otro tipo que se utiliza para la serialización. El nivel de presentación nunca envía o recibe directamente variables de consulta a o desde el nivel intermedio. Los dos niveles intercambian valores, objetos y colecciones de datos concretos. Después de haber recibido una colección, el nivel de presentación puede utilizar LINQ to Objects para consultarla si es necesario.
Al insertar datos, el nivel de presentación puede construir un nuevo objeto y enviarlo al nivel intermedio, o puede hacer que el nivel intermedio construya el objeto en función de los valores que proporciona. En general, la recuperación e inserción de datos en aplicaciones de n niveles no difiere mucho del proceso para aplicaciones de 2 niveles. Para obtener más información, consulte Consulta de la base de datos y Realización y envío de cambios de datos.
Seguimiento de cambios para actualizaciones y eliminaciones
LINQ to SQL admite simultaneidad optimista basada en marcas de tiempo (también denominadas RowVersions) y en valores originales. Si las tablas de base de datos tienen marcas de tiempo, entonces las actualizaciones y eliminaciones requieren poco trabajo adicional en el nivel intermedio o en el nivel de presentación. Sin embargo, si debe utilizar valores originales para las comprobaciones de simultaneidad optimista, entonces el nivel de presentación es el responsable de realizar el seguimiento de esos valores y devolverlos cuando realiza las actualizaciones. Esto es debido a que en el nivel intermedio no se hace un seguimiento de los cambios que se realizan en entidades en el nivel de presentación. De hecho, la recuperación original de una entidad y su posible actualización se realizan generalmente mediante dos instancias completamente independientes de DataContext.
Cuanto mayor es el número de cambios que realiza el nivel de presentación, más complejo se hace realizar el seguimiento de esos cambios y empaquetarlos de vuelta hacia el nivel intermedio. La implementación de un mecanismo para comunicar los cambios depende completamente de la aplicación. El único requisito es que se deben dar a LINQ to SQL aquellos valores originales que se requieren para las comprobaciones de la simultaneidad optimista.
Para obtener más información, vea Recuperación de datos y operaciones CUD en aplicaciones de n niveles (LINQ to SQL).