在前两篇中,我们讲了在SL端通过Entity Framework和WCF Ria Services实现联表查询,其中最关键的是为实体类中的相关属性设定为Include特性,如下:
遗憾的是,这样的实体Include只能在单个域服务间共享实体。如果你尝试撰写多个域服务,并用到关联实体,会报如下错误:
The entity type
'MiniNWModel.Entities.Product'
is
exposed
by
multiple DomainService types.
|
Entity types cannot be shared across DomainServices. SilverlightApplicationSample
|
有人指出,域服务上下文应该是long life的,所以整个应用程序应该只有一个域服务。不过,这难道不是有点扯吗?应用程序所支撑的数据库动辄上百张表,不同的业务范畴建立多个域服务基本是必须的。
一:数据库支撑
本篇所采用的数据库来自于《Entity Framework 4.1 and Poco 使用存储过程联表查询》。
二:数据实体
本篇共涉及两个实体,要让两个实体在多个域服务间共享彼此,必须依赖特性ExternalReference。以下是主表实体:
以下是从表实体:
一定要注意正确匹配Association,否则关联数据的时候会不正确。
三:域服务
两个域服务类没有任何特殊之处。唯一需要注意的是,在调试的时候我们会发现EF联表查询中得到的数据,如果不做特殊处理,从表数据全部会丢失。如下图源码:
在方法GetCategoryWithProductsWithID中,我们会得到从表的数据,这是EF为我们得到的,但是千万不要以为数据会被Ria Service带到SL客户端。
四:SL获取主从表数据
如果SL的某个功能需要同时得到主从表数据,我们必须同时提供包含这两个实体的域服务,在下图代码中,我们首先创建好这两个域服务(图中1)。然后,首先我们必须获取主表数据(图中2),其次是从表数据(图中3),最后,SL客户端要指定两者的关联(图中4):
获取主从表的全部代码如下:
public
class
PrinSubVm : NotificationObject
{
public
DomainServiceCategory DomainServiceCategory {
get
;
set
; }
public
DomainServiceProduct DomainServiceProduct {
get
;
set
; }
private
IList<Category> categoryWithProducts;
public
IList<Category> CategoryWithProducts
{
get
{
return
categoryWithProducts; }
set
{ categoryWithProducts = value;
this
.RaisePropertyChanged<IList<Category>>(() =>
this
.CategoryWithProducts); }
}
public
PrinSubVm()
{
DomainServiceCategory =
new
Web.DomainServiceCategory();
DomainServiceProduct =
new
Web.DomainServiceProduct();
//获取CategoryID(cid)为1的目录
DomainServiceCategory.Load<Category>(DomainServiceCategory.GetCategoryWithProductsWithIDQuery(1),
new
Action<System.ServiceModel.DomainServices.Client.LoadOperation<Category>>(
this
.GetCategoryWithProductsWithIDCallBack),
null
);
//获取CategoryID(cid)为1的目录下的商品
DomainServiceProduct.Load<Product>(DomainServiceProduct.GetProductsByCategoryIDQuery(1),
new
Action<System.ServiceModel.DomainServices.Client.LoadOperation<Product>>(
this
.GetProductsByCategoryIDCallBack),
null
);
//为多个域之间共享实体
DomainServiceCategory.AddReference(
typeof
(Product), DomainServiceProduct);
}
void
GetCategoryWithProductsWithIDCallBack(LoadOperation<Category> arg)
{
CategoryWithProducts = arg.Entities
as
IList<Category>;
}
void
GetProductsByCategoryIDCallBack(LoadOperation<Product> arg)
{
//ProductAndCategorys = arg.Entities as IList<Product>;
}
}
|
有一点我们必须注意,如果要获取从表的数据,仅获取需要的从表记录就可以了,不要加载全部记录,想想那些动辄几百万记录的业务表。SL客户端会自动根据实体的KEY值去关联。
反过来,我们也可以实现从表关联主表。在这里就不一一举例了。但是最后的UI可以作为演示。
本文转自最课程陆敏技博客园博客,原文链接:http://www.cnblogs.com/luminji/archive/2011/07/05/2098210.html,如需转载请自行联系原作者