Implementar asociaciones (EDM)
En el Entity Data Model (EDM), las asociaciones se definen en el lenguaje de definición de esquemas conceptuales (CSDL, Conceptual Schema Definition Language). La implementación de asociaciones asigna las definiciones del CSDL a los metadatos de almacenamiento y crea las asociaciones con el mismo esquema que define las entidades que asocian.
El modo en que las asociaciones se asignan al almacenamiento y se inicializan en el código, y el modo en que una aplicación navega por los datos de las asociaciones se trata en este tema y en el tema Código de una aplicación que usa asociaciones (EDM) a través del examen de un ejemplo completo sencillo.
Asociación de cliente, pedido y línea de pedido
En las aplicaciones de línea de negocio (LOB, Line Of Business), las entidades que representan los clientes y los pedidos están relacionadas lógicamente. Los productos que compran los clientes se representan mediante pedidos. Los clientes y los pedidos se relacionan con el elemento Association denominado Order_Customer
.
Cada entidad de la asociación se conoce como extremo. En esta asociación, el extremo de la misma que representa un cliente se puede relacionar con muchos pedidos, pero el que representa a los pedidos sólo se puede relacionar con un cliente. En la sintaxis del esquema, esto se define mediante el atributo Multiplicity. El extremo de esta asociación denominado Customers
tiene una Multiplicity="1"
y el extremo denominado Orders
tiene una Multiplicity="*"
. Este es un ejemplo de asociación uno a varios. Para obtener más información acerca de los tipos de asociación, vea Asociación (EDM).
Una vez definida una asociación Order_Customer
usando los elementos de esquema de Association, la asociación se incluye como un AssociationSet en un EntityContainer junto con las entidades usadas en este ejemplo. Para obtener más información acerca de los conjuntos de asociaciones y los contenedores de entidades, vea Elemento EntityContainer (CSDL).
En el ejemplo de esquema siguiente se definen las entidades que representan Customers
, Orders
y OrderLines
, y dos asociaciones: Order_Customer
y OrderLine_Order
.
<?xml version="1.0" encoding="utf-8"?>
<Schema Namespace="OrderInfoModel" Alias="Self"
xmlns="https://schemas.microsoft.com/ado/2006/04/edm">
<EntityContainer Name="OrderInfoEntities">
<EntitySet Name="Customers"
EntityType="OrderInfoModel.Customers" />
<EntitySet Name="OrderLines"
EntityType="OrderInfoModel.OrderLines" />
<EntitySet Name="Orders"
EntityType="OrderInfoModel.Orders" />
<AssociationSet Name="Order_Customer"
Association="OrderInfoModel.Order_Customer">
<End Role="Customers" EntitySet="Customers" />
<End Role="Orders" EntitySet="Orders" />
</AssociationSet>
<AssociationSet Name="OrderLine_Order"
Association="OrderInfoModel.OrderLine_Order">
<End Role="Orders" EntitySet="Orders" />
<End Role="OrderLines" EntitySet="OrderLines" />
</AssociationSet>
</EntityContainer>
<EntityType Name="Customers">
<Key>
<PropertyRef Name="CustomerId" />
</Key>
<Property Name="CustomerId" Type="Guid" Nullable="false" />
<Property Name="Name" Type="String" Nullable="false"
MaxLength="50" Unicode="true" FixedLength="false" />
<Property Name="Address" Type="String" Nullable="false"
MaxLength="50" Unicode="true" FixedLength="false" />
<Property Name="City" Type="String" Nullable="false"
MaxLength="50" Unicode="true" FixedLength="false" />
<Property Name="Phone" Type="String" Nullable="false"
MaxLength="50" Unicode="true" FixedLength="false" />
<Property Name="ZipCode" Type="Int32" Nullable="false" />
<NavigationProperty Name="Orders"
Relationship="OrderInfoModel.Order_Customer"
FromRole="Customers" ToRole="Orders" />
</EntityType>
<EntityType Name="OrderLines">
<Key>
<PropertyRef Name="OrderLineId" />
</Key>
<Property Name="OrderLineId" Type="Guid" Nullable="false" />
<Property Name="ProductName" Type="String" Nullable="false"
MaxLength="50" Unicode="true" FixedLength="false" />
<Property Name="Quantity" Type="Int32" Nullable="false" />
<Property Name="UnitPrice" Type="Decimal" Nullable="false"
Precision="19" Scale="4" />
<Property Name="ExtendedPrice" Type="Decimal"
Nullable="false" Precision="19" Scale="4" />
<NavigationProperty Name="Orders"
Relationship="OrderInfoModel.OrderLine_Order"
FromRole="OrderLines" ToRole="Orders" />
</EntityType>
<EntityType Name="Orders">
<Key>
<PropertyRef Name="OrderId" />
</Key>
<Property Name="OrderId" Type="String" Nullable="false"
MaxLength="50" Unicode="true" FixedLength="false" />
<Property Name="TotalAmount"
Type="Decimal" Precision="19" Scale="4" />
<Property Name="Tax" Type="Decimal"
Precision="19" Scale="4" />
<Property Name="ShippingAddress" Type="String"
MaxLength="50" Unicode="true" FixedLength="false" />
<NavigationProperty Name="Customers"
Relationship="OrderInfoModel.Order_Customer"
FromRole="Orders" ToRole="Customers" />
<NavigationProperty Name="OrderLines"
Relationship="OrderInfoModel.OrderLine_Order"
FromRole="Orders" ToRole="OrderLines" />
</EntityType>
<Association Name="Order_Customer">
<End Role="Customers"
Type="OrderInfoModel.Customers" Multiplicity="1" />
<End Role="Orders"
Type="OrderInfoModel.Orders" Multiplicity="*" />
</Association>
<Association Name="OrderLine_Order">
<End Role="Orders"
Type="OrderInfoModel.Orders" Multiplicity="1" />
<End Role="OrderLines"
Type="OrderInfoModel.OrderLines" Multiplicity="*" />
</Association>
</Schema>
La asociación Order_Customer
, NavigationProperty se ha implementado en la entidad Customers
para facilitar la navegación e inicialización de la asociación. La entidad Orders
también especifica una NavigationProperty que se usa para inicializar las OrderLines
que contiene y navegar por ellas. Para obtener más información acerca de la definición de NavigationProperty, vea Propiedades de navegación (EDM).
Asignación y metadatos de las asociaciones
La asignación de Customers
y Orders
de asociaciones uno a varios, como la asociación Order_Customer
, se lleva a cabo en este ejemplo asignando la asociación a una relación de clave externa entre las tablas Orders
y Customers
de una base de datos. Mediante este método, la tabla Orders
contiene varias instancias de la clave externa Customers
para representar los pedidos asociados a cada cliente.
Esquema de almacenamiento
El siguiente esquema del lenguaje de definición de esquemas de almacenamiento (SSDL, Store Schema Definition Language) constituye los metadatos de almacenamiento que representan las tablas Customers
, Orders
y OrderLines
. El esquema SSDL declara estas tablas usando elementos EntityType correspondientes a las tablas de la base de datos. Los tipos de las propiedades de las entidades se asignan según los tipos de datos del sistema de administración de base de datos. Por ejemplo, el tipo de la propiedad Name de la entidad Customers
es del tipo nvarchar en lugar del tipo String que se usa en el esquema del CSDL.
<?xml version="1.0" encoding="utf-8"?>
<Schema Namespace="OrderInfoModel.Store"
Alias="Self" Provider="System.Data.SqlClient"
ProviderManifestToken="2005"
xmlns:store="https://schemas.microsoft.com/ado/2007/12/edm/EntityStoreSchemaGenerator"
xmlns="https://schemas.microsoft.com/ado/2006/04/edm/ssdl">
<EntityContainer Name="OrderInfoModelStoreContainer">
<EntitySet Name="Customers"
EntityType="OrderInfoModel.Store.Customers"
store:Type="Tables" Schema="dbo" />
<EntitySet Name="OrderLines"
EntityType="OrderInfoModel.Store.OrderLines"
store:Type="Tables" Schema="dbo" />
<EntitySet Name="Orders"
EntityType="OrderInfoModel.Store.Orders"
store:Type="Tables" Schema="dbo" />
<AssociationSet Name="Order_Customer"
Association="OrderInfoModel.Store.Order_Customer">
<End Role="Customers" EntitySet="Customers" />
<End Role="Orders" EntitySet="Orders" />
</AssociationSet>
<AssociationSet Name="OrderLine_Order"
Association="OrderInfoModel.Store.OrderLine_Order">
<End Role="Orders" EntitySet="Orders" />
<End Role="OrderLines" EntitySet="OrderLines" />
</AssociationSet>
</EntityContainer>
<EntityType Name="Customers">
<Key>
<PropertyRef Name="CustomerId" />
</Key>
<Property Name="CustomerId"
Type="uniqueidentifier" Nullable="false" />
<Property Name="Name"
Type="nvarchar" Nullable="false" MaxLength="50" />
<Property Name="Address"
Type="nvarchar" Nullable="false" MaxLength="50" />
<Property Name="City"
Type="nvarchar" Nullable="false" MaxLength="50" />
<Property Name="Phone"
Type="nvarchar" Nullable="false" MaxLength="50" />
<Property Name="ZipCode"
Type="int" Nullable="false" />
</EntityType>
<EntityType Name="OrderLines">
<Key>
<PropertyRef Name="OrderLineId" />
</Key>
<Property Name="OrderLineId"
Type="uniqueidentifier" Nullable="false" />
<Property Name="OrderId"
Type="nvarchar" Nullable="false" MaxLength="50" />
<Property Name="ProductName"
Type="nvarchar" Nullable="false" MaxLength="50" />
<Property Name="Quantity" Type="int" Nullable="false" />
<Property Name="UnitPrice"
Type="money" Nullable="false" />
<Property Name="ExtendedPrice"
Type="money" Nullable="false" />
</EntityType>
<EntityType Name="Orders">
<Key>
<PropertyRef Name="OrderId" />
</Key>
<Property Name="OrderId"
Type="nvarchar" Nullable="false" MaxLength="50" />
<Property Name="Customer"
Type="uniqueidentifier" Nullable="false" />
<Property Name="TotalAmount" Type="money" />
<Property Name="Tax"
Type="money" />
<Property Name="ShippingAddress"
Type="nvarchar" MaxLength="50" />
</EntityType>
<Association Name="Order_Customer">
<End Role="Customers"
Type="OrderInfoModel.Store.Customers" Multiplicity="1" />
<End Role="Orders"
Type="OrderInfoModel.Store.Orders" Multiplicity="*" />
<ReferentialConstraint>
<Principal Role="Customers">
<PropertyRef Name="CustomerId" />
</Principal>
<Dependent Role="Orders">
<PropertyRef Name="Customer" />
</Dependent>
</ReferentialConstraint>
</Association>
<Association Name="OrderLine_Order">
<End Role="Orders"
Type="OrderInfoModel.Store.Orders" Multiplicity="1" />
<End Role="OrderLines"
Type="OrderInfoModel.Store.OrderLines" Multiplicity="*" />
<ReferentialConstraint>
<Principal Role="Orders">
<PropertyRef Name="OrderId" />
</Principal>
<Dependent Role="OrderLines">
<PropertyRef Name="OrderId" />
</Dependent>
</ReferentialConstraint>
</Association>
</Schema>
Este esquema de almacenamiento, escrito en SSDL, especifica los tipos de datos para las entidades Orders
y Customers
como se implementan en la base de datos. Las propiedades clave que eran de Type="Guid"
en el esquema CSDL tienen Type="uniqueidentifier"
en la base de datos y se deben especificar utilizando los tipos de base de datos en el esquema de almacenamiento.
A las propiedades que eran de tipo String en CSDL se les asignan tipos nvarchar en el almacenamiento.
Las especificaciones de asociaciones en el esquema de almacenamiento son las mismas que en el esquema del CSDL. Como en el esquema CSDL, las asociaciones de los elementos End se declaran con los atributos Role, Type y Multiplicity. Las asignaciones de End y Role son las mismas que en el CSDL.
El atributo ReferentialConstraint indica que las asociaciones dependen de las estructuras de base de datos. ReferentialConstraint especifica un Principle Role y Dependent Role, y los elementos PropertyRef. Los atributos PropertyRef especifican las propiedades de las entidades End que representan las columnas de clave externa y clave principal de las tablas de base de datos que corresponden a las entidades de la Asociación. El atributo PropertyRef de Principle Role especifica la columna que contiene la clave principal. Por ejemplo, en la asociación Order_Customer
, la PropertyRef de Principle Role es la propiedad CustomerId
. Esta propiedad del esquema de SSDL representa la columna CustomerId
de la tabla Customers
que se asignará a la columna de clave externa Customer
en la tabla Orders
. La clave externa se representa mediante la PropertyRef del Dependent Role.
Especificación de asignaciones
La asignación de asociaciones es similar a la asignación de entidades.
El segmento del archivo del lenguaje de especificación de asignaciones (MSL) siguiente muestra la AssociationSetMapping denominada Order_Customer
. La asociación OrderInfoModel.Order_Customer
especificada en el esquema del CSDL se asigna, en este ejemplo, a la tabla Orders
del esquema del SSDL que representa la base de datos.
La PropertyRef en el lado Customers
de la asociación asigna explícitamente la propiedad CustomerId
de la entidad Customers
a la columna Customer
de la tabla Orders
; esta columna contiene la clave externa que representa la relación entre las tablas Orders
y Customers
de la base de datos.
En el ejemplo siguiente se muestra el esquema de MSL completo del que se ha extraído el segmento anterior. Esta asignación incluye los elementos EntitySetMapping y AssociationSetMapping necesarios para este ejemplo.
<?xml version="1.0" encoding="utf-8"?>
<Mapping Space="C-S"
xmlns="urn:schemas-microsoft-com:windows:storage:mapping:CS">
<EntityContainerMapping
StorageEntityContainer="OrderInfoModelStoreContainer"
CdmEntityContainer="OrderInfoEntities">
<EntitySetMapping Name="Customers">
<EntityTypeMapping
TypeName="IsTypeOf(OrderInfoModel.Customers)">
<MappingFragment StoreEntitySet="Customers">
<ScalarProperty Name="CustomerId"
ColumnName="CustomerId" />
<ScalarProperty Name="Name" ColumnName="Name" />
<ScalarProperty Name="Address" ColumnName="Address" />
<ScalarProperty Name="City" ColumnName="City" />
<ScalarProperty Name="Phone" ColumnName="Phone" />
<ScalarProperty Name="ZipCode" ColumnName="ZipCode" />
</MappingFragment>
</EntityTypeMapping>
</EntitySetMapping>
<EntitySetMapping Name="OrderLines">
<EntityTypeMapping
TypeName="IsTypeOf(OrderInfoModel.OrderLines)">
<MappingFragment StoreEntitySet="OrderLines">
<ScalarProperty Name="OrderLineId"
ColumnName="OrderLineId" />
<ScalarProperty Name="ProductName"
ColumnName="ProductName" />
<ScalarProperty Name="Quantity"
ColumnName="Quantity" />
<ScalarProperty Name="UnitPrice"
ColumnName="UnitPrice" />
<ScalarProperty Name="ExtendedPrice"
ColumnName="ExtendedPrice" />
</MappingFragment>
</EntityTypeMapping>
</EntitySetMapping>
<EntitySetMapping Name="Orders">
<EntityTypeMapping
TypeName="IsTypeOf(OrderInfoModel.Orders)">
<MappingFragment StoreEntitySet="Orders">
<ScalarProperty Name="OrderId" ColumnName="OrderId" />
<ScalarProperty Name="TotalAmount"
ColumnName="TotalAmount" />
<ScalarProperty Name="Tax" ColumnName="Tax" />
<ScalarProperty Name="ShippingAddress"
ColumnName="ShippingAddress" />
</MappingFragment>
</EntityTypeMapping>
</EntitySetMapping>
<AssociationSetMapping Name="Order_Customer"
TypeName="OrderInfoModel.Order_Customer"
StoreEntitySet="Orders">
<EndProperty Name="Customers">
<ScalarProperty Name="CustomerId"
ColumnName="Customer" />
</EndProperty>
<EndProperty Name="Orders">
<ScalarProperty Name="OrderId" ColumnName="OrderId" />
</EndProperty>
</AssociationSetMapping>
<AssociationSetMapping Name="OrderLine_Order"
TypeName="OrderInfoModel.OrderLine_Order"
StoreEntitySet="OrderLines">
<EndProperty Name="Orders">
<ScalarProperty Name="OrderId"
ColumnName="OrderId" />
</EndProperty>
<EndProperty Name="OrderLines">
<ScalarProperty Name="OrderLineId"
ColumnName="OrderLineId" />
</EndProperty>
</AssociationSetMapping>
</EntityContainerMapping>
</Mapping>
Los segmentos de código que se muestran en este tema incluyen los esquemas completos para crear un modelo de objetos de programación en el espacio de nombres OrderInfo
y asignarlo al almacenamiento. Para obtener ejemplos del código que usa este modelo, vea Código de una aplicación que usa asociaciones (EDM).
Métodos de clases parciales
En el EDM, los métodos auxiliares se pueden implementar en clases parciales para mejorar la funcionalidad de las aplicaciones. El siguiente ejemplo es un método auxiliar de la clase Orders
. Este método auxiliar suma los importes ExtendedPrice
de las OrderLines
de un pedido para obtener el TotalAmount
de un Order
. Para obtener más información, vea Métodos auxiliares (EDM).
using System;
using System.Data;
namespace OrderInfoModel
{
public partial class Orders :
global::System.Data.Objects.DataClasses.EntityObject
{
public decimal ComputeOrder()
{
this.TotalAmount = 0;
foreach (OrderLines orderLine in this.OrderLines)
{
orderLine.ExtendedPrice =
orderLine.Quantity * orderLine.UnitPrice;
this.TotalAmount = this.TotalAmount +
orderLine.ExtendedPrice;
}
this.Tax = Decimal.Round(((decimal)this.TotalAmount *
(decimal) .08), 2);
this.TotalAmount = this.TotalAmount + this.Tax;
return (decimal)this.TotalAmount;
}
}
}
Implementar la base de datos
El siguiente script se puede usar para crear la base de datos de este ejemplo. Para crear la base de datos OrderInfo con SQL Server Management Studio:
En el menú Archivo, haga clic en Nuevo y, a continuación, en Consulta de motor de base de datos.
En el cuadro de diálogo Conectarse al motor de base de datos, escriba el host local o el nombre de la instancia de SQL Server, y haga clic en Conectar.
Pegue el siguiente script de Transact-SQL en la ventana de consulta y, a continuación, haga clic en Ejecutar.
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
USE [master]
GO
IF EXISTS (SELECT * FROM sys.databases
WHERE name = 'OrderInfo')
DROP DATABASE OrderInfo;
GO
-- Create the database.
CREATE DATABASE OrderInfo;
GO
USE OrderInfo;
GO
IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[Customers]') AND type in (N'U'))
BEGIN
CREATE TABLE [Customers](
[CustomerId] [uniqueidentifier] NOT NULL,
[Name] [nvarchar](50) NOT NULL,
[Address] [nvarchar](50) NOT NULL,
[City] [nvarchar](50) NOT NULL,
[Phone] [nvarchar](50) NOT NULL,
[ZipCode] [int] NOT NULL,
CONSTRAINT [PK_Customers] PRIMARY KEY CLUSTERED
(
[CustomerId] ASC
)WITH (PAD_INDEX = OFF, IGNORE_DUP_KEY = OFF) ON [PRIMARY]
) ON [PRIMARY]
END
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[Orders]') AND type in (N'U'))
BEGIN
CREATE TABLE [Orders](
[OrderId] [nvarchar](50) NOT NULL,
[Customer] [uniqueidentifier] NOT NULL,
[TotalAmount] [money] NULL,
[Tax] [money] NULL,
[ShippingAddress] [nvarchar](50) NULL,
CONSTRAINT [PK_Orders] PRIMARY KEY CLUSTERED
(
[OrderId] ASC
)WITH (PAD_INDEX = OFF, IGNORE_DUP_KEY = OFF) ON [PRIMARY]
) ON [PRIMARY]
END
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[OrderLines]') AND type in (N'U'))
BEGIN
CREATE TABLE [OrderLines](
[OrderLineId] [uniqueidentifier] NOT NULL,
[OrderId] [nvarchar](50) NOT NULL,
[ProductName] [nvarchar](50) NOT NULL,
[Quantity] [int] NOT NULL,
[UnitPrice] [money] NOT NULL,
[ExtendedPrice] [money] NOT NULL,
CONSTRAINT [PK_OrderLines] PRIMARY KEY CLUSTERED
(
[OrderLineId] ASC
)WITH (PAD_INDEX = OFF, IGNORE_DUP_KEY = OFF) ON [PRIMARY]
) ON [PRIMARY]
END
GO
IF NOT EXISTS (SELECT * FROM sys.foreign_keys WHERE object_id = OBJECT_ID(N'[Order_Customer]') AND parent_object_id = OBJECT_ID(N'[Orders]'))
ALTER TABLE [Orders] WITH CHECK ADD CONSTRAINT [Order_Customer] FOREIGN KEY([Customer])
REFERENCES [Customers] ([CustomerId])
GO
ALTER TABLE [Orders] CHECK CONSTRAINT [Order_Customer]
GO
IF NOT EXISTS (SELECT * FROM sys.foreign_keys WHERE object_id = OBJECT_ID(N'[OrderLine_Order]') AND parent_object_id = OBJECT_ID(N'[OrderLines]'))
ALTER TABLE [OrderLines] WITH CHECK ADD CONSTRAINT [OrderLine_Order] FOREIGN KEY([OrderId])
REFERENCES [Orders] ([OrderId])
GO
ALTER TABLE [OrderLines] CHECK CONSTRAINT [OrderLine_Order]
Vea también
Conceptos
Asociación (EDM)
Propiedades de navegación (EDM)
Implementar entidades (EDM)
Otros recursos
Especificación de asignaciones y esquemas (Entity Framework)
Aplicaciones de ejemplo (Entity Framework)