关系与导航属性

本主题概述实体框架如何管理实体间的关系。还对如何映射和操作关系提供了一些指南。

关系、导航属性和外键

在关系数据库中,表之间的关系(也称为关联)是通过外键定义的。外键 (FK) 是用于在两个表的数据之间建立并强制链接的一列或列组合。有三种关系类型:一对一、一对多和多对多。在一对多关系中,外键是在表示关系多端的表上定义的。 多对多关系涉及定义第三个表(也称为接合或联接表),主键由来自两个相关表的外键组成。在一对一关系中,主键还用作外键,两个表都没有单独的外键列。

下图显示的两个表存在一对多关系。Course 表为依赖表,因为它包含 DepartmentID 列,该列链接到 Department 表。

在实体框架中,实体可以通过关联(关系)与其他实体相关。每个关系都包含两端,它们描述关系中两个实体的实体类型以及类型的多重性(一、零 或一、多)。关系可由引用约束控制,该引用约束描述了关系中的哪端为 Principal Role 以及哪端为 Dependent Role。

导航属性为在两个实体类型间导航关联提供了一种方式。针对对象参与到其中的每个关系,各对象均可以具有导航属性。使用导航属性,可以在两个 方向上导航和管理关系,返回引用对象(如果多重性为一或者零或一)或集合(如果多重性为多)。也可以选择使用单向导航,在这种情况下,只对参与关系的一种 而不是两种类型定义导航属性。

建议在映射到数据库中外键的模型中包含属性。加入了外键属性,您就可以通过修改依赖对象的外键值来创建或更改关系。此类关联称为外键关联。 在 N 层应用程序中,使用外键更为重要。请注意,在 1 对 1 或 1 对 0..1 关系中,没有单独的外键列,主键属性用作外键并且始终包含在模型中。

当模型中不包含外键列时,关联信息将作为独立对象管理。关系是通过对象引用而不是外键属性跟踪的。这种关联类型称为“独立关联”。修改独立关联 的最常见方式是修改为参与关联的每个实体生成的导航属性。

可以在您的模型中选择使用一种或两种类型的关联。不过,如果多对多关系是通过只包含外键的联接表连接的纯粹关系,EF 将使用独立关联来管理这样的多对多关系。

下图所示为使用实体框架设计器创建的概念模型。该模型包含两个参与一对多关系的实体。这两个实体都有导航属性。Course 为依赖实体,它定义了 DepartmentID 外键属性。

下图所示为使用 Code First 创建的相同模型。

配置/映射关系

本页面的其余部分介绍如何使用关系访问和操作数据。有关在模型中如何设置关系的信息,请参见以下页面。

创建和修改关系

外键关联 中,更改关系时,具有 EntityState.Unchanged 状态的依赖对象的状态会改为 EntityState.Modified。在独立关系中,更改关系不会更新依赖对象的状态。

下面的示例演示如何使用外键属性和导航属性关联相关对象。使用外键关联时可以使用任一种方法更改、创建或修改关系。使用独立关联,则不能使用外键属性。

  • 通过为外键属性指定新值,如下例所示。

    course.DepartmentID = newCourse.DepartmentID;

  • 下面的代码通过将外键设置为 null 删除了关系。请注意,外键属性必须不可为 Null。

    course.DepartmentID = null;

    注意:
    果引用处于已添加状态(在本例中为 course 对象),在调用 SaveChanges
    之前,引用导航属性将不与新对象的键值同步。由于对象上下文在键值保存前不包含已添加对象的永久键,因此不发生同步。如果必须在设置关系时使新对象完全同
    步,请使用以下方法中的一种。

  • 通过将一个新对象分配给导航属性。下面的代码在 course 和 department 之间创建关系。如果对象附加到上下文,course 也会添加到 department.Courses 集合中,course 对象的相应的外键属性设置为 department 的键属性值。

    course.Department = department;

  • 要删除该关系,请将导航属性设置为 null。如果使用的是基于 .NET 4.0 的实体框架,则需要先加载相关端,然后再将其设置为 Null。例如:

    context.Entry(course).Reference(c => c.Department).Load();
    course.Department = null;

    从实体框架 5.0(它基于 .NET 4.5)开始,不必加载相关端就可以将关系设置为 Null。也可以使用以下方法将当前值设置为 Null。

    context.Entry(course).Reference(c => c.Department).CurrentValue = null;

  • 通过在实体集合中删除或添加对象。例如,可以将 Course 类型的对象添加到 department.Courses 集合中。此操作将在特定 course 和特定 department 之间创建关系。如果对象附加到上下文,course 对象的 department 引用和外键属性将设置为相应的 department

    department.Courses.Add(newCourse);

  • 通过使用 ChangeRelationshipState 方法更改两个实体对象间指定关系的状态。此方法是处理 N 层应用程序和独立关联 时最常用的方法(不能用于外键关联)。此外,要使用此方法,必须下拉到 ObjectContext,如下例所示。

    在下面的示例中,Instructor 和 Course 之间存在多对多关系。调用 ChangeRelationshipState 方法并传递 EntityState.Added 参数,使 SchoolContext 知道在这两个对象间添加了关系。

           ((IObjectContextAdapter)context).ObjectContext.
                     ObjectStateManager.
                      ChangeRelationshipState(course, instructor, c => c.Instructor, EntityState.Added);

    请注意,如果是更新(而不仅是添加)关系,添加新关系后必须删除旧关系:

           ((IObjectContextAdapter)context).ObjectContext.
                      ObjectStateManager.
                      ChangeRelationshipState(course, oldInstructor, c => c.Instructor, EntityState.Deleted);    

同步 FK 和导航属性之间的更改

使用上述方法中的一种更改附加到上下文的对象的关系时,实体框架需要保持外键、引用和集合同步。实体框架使用代理自动管理 POCO 实体的这种同步(也称为关系修复)。有关更多信息,请参见使用代理

如果不通过代理使用 POCO 实体,则必须确保调用 DetectChanges 方法同步上下文中的相关对象。请注意,下面的 API 会自动触发 DetectChanges 调用。

  • DbSet.Add
  • DbSet.Find
  • DbSet.Remove
  • DbSet.Local
  • DbContext.SaveChanges
  • DbSet.Attach
  • DbContext.GetValidationErrors
  • DbContext.Entry
  • DbChangeTracker.Entries
  • 对 DbSet 执行 LINQ 查询

加载相关对象

在实体框架中,一般使用导航属性加载与定义的关联返回的实体相关的实体。有关更多信息,请参见加载相关对象


注意:在外键关联中,加载依赖对象的相关端时,将根据内存中当前的相关外键值加载相关对象:

 
// Get the course where currently DepartmentID = 1.
Course course2 = context.Courses.First(c=>c.DepartmentID == 2); // Use DepartmentID foreign key property
// to change the association.
course2.DepartmentID = 3; // Load the related Department where DepartmentID = 3
context.Entry(course).Reference(c => c.Department).Load();

在独立关联中,基于当前数据库中的外键值查询依赖对象的相关端。不过,如果修改了关系,并且依赖对象的引用属性指向对象上下文中加载的不同主对象,实体框架将尝试创建关系,就像它在客户端定义的那样。

管理并发

在外键和独立关联中,并发检查都是基于实体键和模型中定义的其他实体属性的。使用 EF 设计器创建模型时,将 ConcurrencyMode 特性设置为 fixed 可指定应对该属性进行并发检查。使用 Code First 定义模型时,应对要进行并发检查的属性使用 ConcurrencyCheck 注释。使用 Code First 时,也可以使用 TimeStamp 注释指定应对属性进行并发检查。一个给定类中只能有一个时间戳属性。Code First 将此属性映射到数据库中一个不可为 Null 的字段。

建议始终使用外键关联处理参与并发检查和解析的实体。

有关更多信息,请参见开放式并发模式

使用重叠键

重叠键是指键中某些属性亦是实体中其他键的一部分的那些组合键。在独立关联中不能包含重叠键。若要更改包含重叠键的外键关联,我们建议您修改外键值而不要使用对象引用。

关系与导航属性(摘自微软MSDN)的更多相关文章

  1. Dapper实现一对多对象关系聚合导航属性

    领域对象:Game(游戏), Room(游戏群),两者一对多的关系,SQL语句中会用到JOIN public class Game : AggregateRoot { public string Ta ...

  2. MVC3+EF4.1学习系列(五)----- EF查找导航属性的几种方式

    文章索引和简介 通过上一篇的学习 我们把demo的各种关系终于搭建里起来 以及处理好了如何映射到数据库等问题 但是 只是搭建好了关系 问题还远没有解决 这篇就来写如何查找导航属性 和查找导航属性的几种 ...

  3. FreeSql (十八)导航属性

    导航属性是 FreeSql 的特色功能之一,可通过约定配置.或自定义配置对象间的关系. 导航属性有 OneToMany, ManyToOne, ManyToMany, OneToOne, Parent ...

  4. FreeSql 导航属性的联级保存功能

    写在前面 FreeSql 一个款 .net 平台下支持 .net framework 4.5+..net core 2.1+ 的开源 ORM.单元测试超过3100+,正在不断吸引新的开发者,生命不息开 ...

  5. EntityFramework 6.x和EntityFramework Core关系映射中导航属性必须是public?

    前言 不知我们是否思考过一个问题,在关系映射中对于导航属性的访问修饰符是否一定必须为public呢?如果从未想过这个问题,那么我们接下来来探讨这个问题. EF 6.x和EF Core 何种情况下必须配 ...

  6. 你所不知道的库存超限做法 服务器一般达到多少qps比较好[转] JAVA格物致知基础篇:你所不知道的返回码 深入了解EntityFramework Core 2.1延迟加载(Lazy Loading) EntityFramework 6.x和EntityFramework Core关系映射中导航属性必须是public? 藏在正则表达式里的陷阱 两道面试题,带你解析Java类加载机制

    你所不知道的库存超限做法 在互联网企业中,限购的做法,多种多样,有的别出心裁,有的因循守旧,但是种种做法皆想达到的目的,无外乎几种,商品卖的完,系统抗的住,库存不超限.虽然短短数语,却有着说不完,道不 ...

  7. EF Core反向导航属性解决多对一关系

    多对一是一种很常见的关系,例如:一个班级有一个学生集合属性,同时,班级有班长.语文课代表.数学课代表等单个学生属性,如果定义2个实体类,班级SchoolClass和学生Student,那么,班级Sch ...

  8. Json.net对于导航属性的处理(解决对象循环引用)

    对于两张表A.B多对多的关系中,A的导航属性中有B,B的导航属性中有A,这样Json.net对A或者B对象序列化时会形成死循环 所以对于导航属性要加标签 首先在A.B实体类工程(Model)中引用Js ...

  9. ASP.NET MVC深入浅出(被替换) 第一节: 结合EF的本地缓存属性来介绍【EF增删改操作】的几种形式 第三节: EF调用普通SQL语句的两类封装(ExecuteSqlCommand和SqlQuery ) 第四节: EF调用存储过程的通用写法和DBFirst模式子类调用的特有写法 第六节: EF高级属性(二) 之延迟加载、立即加载、显示加载(含导航属性) 第十节: EF的三种追踪

    ASP.NET MVC深入浅出(被替换)   一. 谈情怀-ASP.NET体系 从事.Net开发以来,最先接触的Web开发框架是Asp.Net WebForm,该框架高度封装,为了隐藏Http的无状态 ...

随机推荐

  1. EF快速开发定义数据接口类(转)

    using System; using System.Linq; using System.Linq.Expressions; using System.Data.Objects; namespace ...

  2. EF的增删改查

    //获取分组信息        public List<UserGroupLogSys> GetUserGroupLogSyslist(int pageIndex, int pageSiz ...

  3. Android系统文件夹组织结构

  4. jQuery原生框架-----------------核心框架

    // 1.自调防止全局变量污染(function( window ) { var version = '1.0.0'; var document = window.document; var arr ...

  5. [置顶] Android 2016新技术

    版权声明:分享技术,传播快乐.如果本博客对你有帮助,请在我的博客首页为我打赏吧! 2016你需要了解Android有以下新兴的技术与框架,有些也许还不成熟,但是你应该去了解下,也许就是未来的方向. K ...

  6. Hadoop基本操作

    命令基本格式: hadoop fs -cmd < args > 1.ls hadoop fs -ls / 列出hdfs文件系统根目录下的目录和文件 hadoop fs -ls -R / 列 ...

  7. 2016 - 2 - 20 ARC知识总结(二 autorelease概念及实现)

    首先祝自己生日快乐~23咯~ 一  autorelease的概念 autorelease会像C语言的自动变量那样来对待对象实例.当超出作用域(相当于变量作用域)时,对象的实例release实力方法被调 ...

  8. sass/less/stylus css编译

    早上来了听一同事说stylus如何才能编译成css文件,瞬时间有点蒙,一听感觉和less是差不多的功能,随着就上网去查,然后发现这个文章,介绍了这三种sass/less/stylus的安装和语法,贴在 ...

  9. 指针数组 null与空字符串

    指针数组常适用于指向若干字符串,这样使字符串处理更加灵活方便. 在c++中,null表示:对象为空,它是对指针而言的.而""表示:值为空,它是对字符串而言的.

  10. myeclipse和eclipse哪个好?

    eclipse是开发java的一款专业IDE,myeclipse本身是eclipse的插件(用于开发 javaee的平台),后来myeclipse干脆把eclipse集成进去了,所以现在你下载一个my ...