对于关系型数据与对象数据之间最大的隔阂就是由标识列连接起来的行(关系型数据)与由集合保存的对象(对象数据)之间的冲突。
例如某个Subject对象(也就是数据库中的Subject表),从Subject对象可以找到属于它的Book集合,甚至可以在Book对象继续找到其包含的Author集合或者其他暴露出的属性,幸运的是LINQ TO SQL提供了足够的支持.
从Subject类的定义中可以看到,它可包含一个名为Books的属性,用来让程序直接访问到该主题下的图书图像。底下代码将Book对象集合以 System.Data.Linq.EntitySet<Book>泛型类型的形式暴露为Subject类的Books属性的方法。
using System.Data.Linq.Mapping;
{
[Table]
Public class Subject
{
[Column(IsPrimaryKey=true, Name="ID")]
public Guid SubjectID{get;set;}
[Column]
public String Description{get;set;}
[Column]
public String Name{get;set;}
[Association(OtherKey="SubjectId")]
public EntitySet<Book>books{get;set;}
}
}
与对待表和列一样,代码中需要使用Association属性明确告知框架这些对象之间的关系。添加关联的前提条件就是了解Book类型和Subject类型之间是怎样建立联系的.Book类包含一个SubjectId属性,已经被映射到了数据库中Book表的Subject列上。这样,对于Subject类的Books属性,我们就应该指定其相关记录的Key为Book类中SubjectId(由OtherKey指定,或相关对象中作为Key的属性)
遍历对象的层次结构
foreach(Subject subject in dataContext.GetTable<Subject>())
{
Console.writeline(subject.Name)
foreach(Book book in subject.Books)
{
Console.writeline("...{0}",book,Title);
}
}
运行该查询时,可以看到其默认的结果与外连接的结果相同,从对象的角度来看,在获取主题集合时,我们并不知道该主题下是否有与其相关的图书。只有在开始遍历其Books属性时才能得到其包含的图书信息,这样就有可能在显示了主题之后,并没有任何的图书包含在其中。同样,若某本图书没有归为任何的主题,那么也不能显示在结果列表中。
若想继续过滤上述结果,我们需要介绍两个新的扩展方法:Any和All.只有两个结果集合中都存着相关的记录,Any方法才会将其返回。这样,若想返回包含有图书的主题集合,即可使用Any扩展方法。
var query=from subject in Subjects where subject.Books.Any() select subject;
如果需要的话,也可以在where字句中通过简单的取反操作得到不包含任何图书的主题集合
var query=from subject in Subjects where !subject.Books.Any() select subject;