在EDM中的关系Assciation 如果是One:One or One:many 按照设计器的Wizard一步步下来,然后做TableMapping就可以了(虽然在EDM之前也接触过Nhibernate,但仅仅局限于对它的查询.mapping的概念的不是很强烈)那么对于many:many的Associationn呢,以Order <- (*-*) -> Product 这样的一个many:many来讲,我们在数据库模型里面必须要借助第三表OrderLines来实现
即Order <- (1-*) -> OrderLines<- (*-1) -> Product在EDM中Order,Product表映射为实体,而OrderLines要怎么处理?
有2中情况:
1.如果当OrderLines表只有2个字段,分别作为外键指向Order,Product,那么在EDM中只需将Association(Order -Product)关系映射到OrderLines(这里使用的表是OrderLines1只有2个字段)
图1
创建一个testProject测试一下
我们可以通过
var query = from oo in context.Orders from pp in oo.Products where oo.ID == 1 select pp;
var query1 = context.Orders.Include("Products").Where("it.ID==1");
var query2 = context.Orders.Include("Products").Where(q => q.ID == 1);
这样的的方式对Order ,Product进行关联查询了.
2.当OrderLines表除了存放Order Products关系外,还有其他字段,这就是所谓的有效负载(Payloads),我们只好将OrderLines也作为对象展现在
EDM中
图2
但是在testMethod2中
这样就很难从Order查询得到Product的集合,反之亦然.那么能不能也像第一种情况那样不将OrderLines映射为一个对象,当然是可以的, 但是必须将OrderLines多余的信息Quantity, PercentDiscount设为空,或设默认值.否则会有如下错误.
但是那样的话就无法访问到OrderLines表中 Quantity 和PercentDiscount了.不过在Alex James的blog 上看到关于AEF开发小组在设计AEF时
对于Association(Many to many )的有效负载(Payloads)问题的考虑,准备通过暴露一个事件来访问 OrderLines(Association),不过由于实现比较复杂
被开发小组否定了,给出了另一种解决方案:创建一个ReadOnly的Association,就可以在EDM中保留OrderLines对象又可以像第一种情况一样
实现Order, Product的相互访问,进而实现从Order,Product的一个loop.
用XML浏览器打开图2的模型,
1.在ssdl内 添加一个entityType: ProductOrders
2.添加一个EntitySet
2 <DefiningQuery>
3 SELECT ProductID, OrderID FROM OrderLines
4 </DefiningQuery>
5 </EntitySet>
3.在csdl内 添加2个many to many 的Association:
4 添加AssociationSet:ProductOrders
5从ssdl 到csdl的映射:
然后在csdl分为Order Product添加Nagivation Property:
<NavigationProperty Name="Products" Relationship="edmModel.ProductOrders" FromRole="Orders" ToRole="Products"/>
<NavigationProperty Name="Orders" Relationship="edmModel.ProductOrders" FromRole="Products" ToRole="Orders">
做完这些保存打开设计器,看完成了一个Loop:
图3
这样就可以在testMethod2中使用这些查询
但对于以上模型,使用下面的o.Products.Add(p)是不能成功的,因为在ssdl中我们只能定义Query .