为ASP.NET MVC应用程序实现继承

这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译,这里是第十一篇:为ASP.NET MVC应用程序实现继承

原文:Implementing Inheritance with the Entity Framework 6 in an ASP.NET MVC 5 Application

译文版权所有,谢绝全文转载——但您可以在您的网站上添加到该教程的链接。

在之前的教程中,您已经学习了如何处理并发异常。在本教程中,我们将介绍如何实现继承。

在面向对象的编程中,你可以使用继承以便重用代码。在本教程中,您将更改Instructor和Student类,使它们从包含姓名属性的Person基类派生。你无须改动任何WEB页面,但你的改动会自动反映在数据库中。

映射继承到数据库的选项

数据模型中的Instructor和Student类有几个相同的属性:

假设您想要通过共享教师和学生实体的属性来消除冗余的代码,或者您想要编写一个无需关心名称是否来自学生或教师的从而正确格式化姓名的服务。你可以创建一个包含这些共享属性的Person基类,然后使教师和学生实体的类从基类继承,如下图所示:

###

在数据库中,这种继承结构有几种表现形式。你可以创建一个Person数据表,包含教师和学生和学生信息的单个表,某些列可能仅适用于教师(雇佣日期),某些只适用于学生(注册日期),某些两者都要使用(姓、名)。通常情况下,你会有一个标识列,以指示每一行所代表的类型,例如,标识列可能使用"Instructor"来表示教师,"Student"来表示学生。

从单个数据库表生成实体继承结构的模式被称为每层一表继承模式。

替代方法是使用看起来更像继承结构的数据库,例如,你可以只在Person表中包含学生和教师共有的属性,将独有的属性放在各自单独的表中。

使每个实体类都建立一个数据库表的模式成为每类型一表继承。

但另一种选择是将所有非抽象类型映射到单个表。所有类别的属性,包括继承的,都将映射到相应表中的列。这种模式被称为每具体类一表继承。如果您实现了Person,Student和Instructor类的具体类一表继承,Student和Instructor数据表将和之前你看到的没有两样。

每具体类一表和每层一表在实体框架中通常会提供比每类型一表更好地性能,因为每类型一表可能会导致复杂的连接查询。

本教程将演示如何实现每层一表继承。每层一表是实体框架默认的继承模式。你所要做的就是创建一个Person类,修改Instructor和Student类派生自Person,将新的类添加到DbContext及创建迁移。

创建Person类

在Models文件夹中,使用下面的代码创建Person类:

using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema; namespace ContosoUniversity.Models
{
public abstract class Person
{
public int ID { get; set; }
[Required]
[Display(Name = "姓")]
[StringLength(50)]
public string LastName { get; set; } [Required]
[Column("FirstName")]
[Display(Name = "名")]
[StringLength(50)]
public string FirstMidName { get; set; } [Display(Name = "全名")]
public string FullName
{
get
{
return LastName + ", " + FirstMidName;
}
}
}
}

使Student和Instructor类继承自Person

在Instructor类中,修改类从Person派生并删除姓名字段,如下面的代码:

using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;namespace ContosoUniversity.Models
{
public class Instructor :Person
{
public int ID { get; set; } [DataType(DataType.Date)]
[DisplayFormat(DataFormatString = "{0:yyyy-MM-dd}",ApplyFormatInEditMode = true)]
[Display(Name = "聘用日期")]
public DateTime HireDate { get; set; } public virtual ICollection<Course> Courses { get; set; }
public virtual OfficeAssignment OfficeAssignment { get; set; }
}
}

同样也对Student类进行修改:

using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations; namespace ContosoUniversity.Models
{
public class Student : Person
{
[Display(Name = "注册日期")]
[DataType(DataType.Date)]
[DisplayFormat(DataFormatString = "{0:yyyy-MM-dd}", ApplyFormatInEditMode = true)]
public DateTime EnrollmentDate { get; set; } public virtual ICollection<Enrollment> Enrollments { get; set; }
}
}

向模型中添加Person实体类型

向SchoolContext.cs中添加一个Person实体的DbSet属性:

        public DbSet<Person> People { get; set; }

就是在实体框架中实现继承做需要的全部修改。稍后您会看到数据库在更新后,会有一个新建的Person数据表。

创建及更新一个迁移文件

在软件包管理器控制台中,输入以下命令:

Add-Migration Inheritance

之后运行update-database命令,命令将失败。因为实体框架不知道如何对我们现有的数据进行迁移,错误消息类似下面这样:

打开Migrations\<时间戳>-Inheritance.cs文件,使用下面的代码替换Up方法:

public override void Up()
{
// Drop foreign keys and indexes that point to tables we're going to drop.
DropForeignKey("dbo.Enrollment", "StudentID", "dbo.Student");
DropIndex("dbo.Enrollment", new[] { "StudentID" }); RenameTable(name: "dbo.Instructor", newName: "Person");
AddColumn("dbo.Person", "EnrollmentDate", c => c.DateTime());
AddColumn("dbo.Person", "Discriminator", c => c.String(nullable: false, maxLength: 128, defaultValue: "Instructor"));
AlterColumn("dbo.Person", "HireDate", c => c.DateTime());
AddColumn("dbo.Person", "OldId", c => c.Int(nullable: true)); // Copy existing Student data into new Person table.
Sql("INSERT INTO dbo.Person (LastName, FirstName, HireDate, EnrollmentDate, Discriminator, OldId) SELECT LastName, FirstName, null AS HireDate, EnrollmentDate, 'Student' AS Discriminator, ID AS OldId FROM dbo.Student"); // Fix up existing relationships to match new PK's.
Sql("UPDATE dbo.Enrollment SET StudentId = (SELECT ID FROM dbo.Person WHERE OldId = Enrollment.StudentId AND Discriminator = 'Student')"); // Remove temporary key
DropColumn("dbo.Person", "OldId"); DropTable("dbo.Student"); // Re-create foreign keys and indexes pointing to new table.
AddForeignKey("dbo.Enrollment", "StudentID", "dbo.Person", "ID", cascadeDelete: true);
CreateIndex("dbo.Enrollment", "StudentID");
}

这段代码执行了下列数据库更新任务:

  • 删除了指向学生数据表的外键约束和索引
  • 重命名Instructor表为Person表并进行了修改:
    • 为学生添加了可以为空的EnrollmentDate
    • 添加了标识列,以指示行是否为学生或教师
    • 使雇佣日期可以为空,因为学生的行不会有雇佣日期
    • 添加一个临时字段用来更新指向学生的外键。当你将学生复制回Person表时他们会有一个新的主键值。
  • 将数据从学生表复制到Person表,这会导致学生有一个新的主键值
  • 修复了指向学生的外键值
  • 重新创建外键约束和索引,现在它们指向Person表

(如果你使用了GUID而不是int作为主键类型,学生的主键值不会改变,上面的几个步骤可能被省略。)

再次运行update-database命令。

注意:您可以仍然得到一个错误,在进行迁移或架构更改时,如果迁移的错误无法解决,您可以通过更改web.config连接字符串或删除该数据库的方法来继续本教程,最简单的方法是重新命名数据库。

测试

运行应用程序,尝试各种操作,一切都正常运行。

在服务器资源管理器中,展开数据连接,展开SchoolContext的数据表,你会看到Person表已经替换了Student和Instructor表,打开Person表,你会看到之前的学生和教师的信息。

下面的关系图说明了新数据库的结构:

部署到Windows Azure

本章跳过……

总结

你现在实现了Person、Student和Instructor类的每层次一个表继承。有关其他继承结构的信息,请参阅TPH Inheritance PatternTPT Inheritance Pattern。在下一教程中,您将看到如何实现仓储和单元工作模式。

作者信息

  Tom Dykstra - Tom Dykstra是微软Web平台及工具团队的高级程序员,作家。

 
 

为ASP.NET MVC应用程序实现继承的更多相关文章

  1. [渣译文] 使用 MVC 5 的 EF6 Code First 入门 系列:为ASP.NET MVC应用程序实现继承

    这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译,这里是第十一篇:为ASP.NET MVC应用程 ...

  2. 使用Metrics.NET 构建 ASP.NET MVC 应用程序的性能指标

    通常我们需要监测ASP.NET MVC 或 Web API 的应用程序的性能时,通常采用的是自定义性能计数器,性能计数器会引发无休止的运维问题(损坏的计数器.权限问题等).这篇文章向你介绍一个新的替代 ...

  3. asp.net MVC 应用程序的生命周期

    下面这篇文章总结了 asp.net MVC 框架程序的生命周期.觉得写得不错,故转载一下. 转载自:http://www.cnblogs.com/yplong/p/5582576.html       ...

  4. [渣译文] 使用 MVC 5 的 EF6 Code First 入门 系列:为ASP.NET MVC应用程序创建更复杂的数据模型

    这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译,这里是第六篇:为ASP.NET MVC应用程序 ...

  5. [渣译文] 使用 MVC 5 的 EF6 Code First 入门 系列:为ASP.NET MVC应用程序处理并发

    这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译,这里是第十篇:为ASP.NET MVC应用程序 ...

  6. 为ASP.NET MVC应用程序使用高级功能

    为ASP.NET MVC应用程序使用高级功能 这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译, ...

  7. ASP.NET MVC应用程序处理并发

    为ASP.NET MVC应用程序处理并发 2014-05-14 08:37 by Bce, 694 阅读, 2 评论, 收藏, 编辑 这是微软官方教程Getting Started with Enti ...

  8. 为ASP.NET MVC应用程序创建更复杂的数据模型

    为ASP.NET MVC应用程序创建更复杂的数据模型 2014-05-07 18:27 by Bce, 282 阅读, 1 评论, 收藏, 编辑 这是微软官方教程Getting Started wit ...

  9. ASP.NET MVC应用程序实现下载功能

    ASP.NET MVC应用程序实现下载功能 上次Insus.NET有在MVC应用程序实现了上传文件的功能<MVC应用程序显示上传的图片> http://www.cnblogs.com/in ...

随机推荐

  1. HDU 5052 Yaoge’s maximum profit 光秃秃的树链拆分 2014 ACM/ICPC Asia Regional Shanghai Online

    意甲冠军: 特定n小点的树权. 以下n每一行给出了正确的一点点来表达一个销售点每只鸡价格的格 以下n-1行给出了树的侧 以下Q操作 Q行 u, v, val 从u走v,程中能够买一个鸡腿,然后到后面卖 ...

  2. JavaEE(6) - JMS消息选择和查看

    1. JMS消息的类型.消息头和消息属性 消息类型: StreamMessage MapMessage TextMessage ObjectMessage BytesMessage JMS消息中的消息 ...

  3. iframe参数

    iframe参数: <iframe src="test.jsp" width="100″ height="50″ frameborder="no ...

  4. 第11章 享元模式(Flyweight Pattern)

    原文 第11章 享元模式(Flyweight Pattern) 概述:   面向对象的思想很好地解决了抽象性的问题,一般也不会出现性能上的问题.但是在某些情况下,对象的数量可能会太多,从而导致了运行时 ...

  5. 远程数据client交换器

    不太繁忙的文本. 要被写入

  6. crawler_大型舆情架构图

  7. Windows RPC

    转载 Windows RPC Demo实现 本文参考并整理以下相关文章 1. <远程过程调用> -百度百科 2. <RPC 编程> -http://www.ibm.com/de ...

  8. 在Magento产品页面的使用jqZoom

    Magento在产品页面提供了一个简单的图片放大功能,不是非常好,假设考虑使用放大镜来使用户查看产品的大图.能够考虑使用基于jQuery的插件,jqZoom便是一款优秀的放大镜插件.以下将介绍如何把j ...

  9. HDU 3683 模拟&amp;搜索

    给出五子棋残局,推断三步内能否分出胜负,玩家为当前该走旗子的颜色,下一步为白棋或黑棋不定. 依照顺序推断就可以: 1:推断棋盘是否合法,并确定玩家颜色 2:推断当前玩家颜色是否有一个必胜点,有玩家则在 ...

  10. Struts2和Struts1的主要区别(完整版)

    Struts1和Struts2的区别和对比: Action 类: • Struts1要求Action类继承一个抽象基类.Struts1的一个普遍问题是使用抽象类编程而不是接口,而struts2的Act ...