为ASP.NET MVC应用程序实现继承
为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 Pattern和TPT Inheritance Pattern。在下一教程中,您将看到如何实现仓储和单元工作模式。
作者信息
Tom Dykstra - Tom Dykstra是微软Web平台及工具团队的高级程序员,作家。
为ASP.NET MVC应用程序实现继承的更多相关文章
- [渣译文] 使用 MVC 5 的 EF6 Code First 入门 系列:为ASP.NET MVC应用程序实现继承
这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译,这里是第十一篇:为ASP.NET MVC应用程 ...
- 使用Metrics.NET 构建 ASP.NET MVC 应用程序的性能指标
通常我们需要监测ASP.NET MVC 或 Web API 的应用程序的性能时,通常采用的是自定义性能计数器,性能计数器会引发无休止的运维问题(损坏的计数器.权限问题等).这篇文章向你介绍一个新的替代 ...
- asp.net MVC 应用程序的生命周期
下面这篇文章总结了 asp.net MVC 框架程序的生命周期.觉得写得不错,故转载一下. 转载自:http://www.cnblogs.com/yplong/p/5582576.html ...
- [渣译文] 使用 MVC 5 的 EF6 Code First 入门 系列:为ASP.NET MVC应用程序创建更复杂的数据模型
这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译,这里是第六篇:为ASP.NET MVC应用程序 ...
- [渣译文] 使用 MVC 5 的 EF6 Code First 入门 系列:为ASP.NET MVC应用程序处理并发
这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译,这里是第十篇:为ASP.NET MVC应用程序 ...
- 为ASP.NET MVC应用程序使用高级功能
为ASP.NET MVC应用程序使用高级功能 这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译, ...
- ASP.NET MVC应用程序处理并发
为ASP.NET MVC应用程序处理并发 2014-05-14 08:37 by Bce, 694 阅读, 2 评论, 收藏, 编辑 这是微软官方教程Getting Started with Enti ...
- 为ASP.NET MVC应用程序创建更复杂的数据模型
为ASP.NET MVC应用程序创建更复杂的数据模型 2014-05-07 18:27 by Bce, 282 阅读, 1 评论, 收藏, 编辑 这是微软官方教程Getting Started wit ...
- ASP.NET MVC应用程序实现下载功能
ASP.NET MVC应用程序实现下载功能 上次Insus.NET有在MVC应用程序实现了上传文件的功能<MVC应用程序显示上传的图片> http://www.cnblogs.com/in ...
随机推荐
- 嘿嘿。今天学习了AJAX的几个方法
原文:嘿嘿.今天学习了AJAX的几个方法 今天学习了AJAX的几个方法,其实我很早在公司实习的时间就认识了它,但是对它一无所知,也并没有去学习它,今天学习它让我感到很兴奋因为重新了解了它,嘿嘿,下面就 ...
- PL/SQL 9 许可证
code:j6stndb9tk72xfbhbqczcdqnjd8lyj466n number:882851 ps:xs374ca 我是PL/SQL版本号是: Version 9.0.3.1641 要注 ...
- hdu 最大三角形(凸包+旋转卡壳)
老师在计算几何这门课上给Eddy布置了一道题目,题目是这样的:给定二维的平面上n个不同的点,要求在这些点里寻找三个点,使他们构成的三角形拥有的面积最大.Eddy对这道题目百思不得其解,想不通用什么方法 ...
- crm使用soap删除实体
//C# 代码: //DeleteEntityRequest request = new DeleteEntityRequest(); //request.LogicalName = "ne ...
- 在vc正在使用xtremetoolkit接口库-----使用简单的控制
首先,我们需要在StdAfx.h增加头文件: #include "XTToolkitPro.h" #include "XTPResource.h" 在test. ...
- 都能看懂的嵌入式linux/android alsa_aplay alsa_amixer命令行使用方法
前几天在嵌入式linux上用到alsa command,网上查的资料多不给力,仅仅有动手一点点查,最终能够用了,将这个用法告诉大家,以免大家少走弯路. 0.先查看系统支持哪几个alsa cmd: ll ...
- 细说 ASP.NET Cache 及其高级用法【转】
阅读目录 开始 Cache的基本用途 Cache的定义 Cache常见用法 Cache类的特点 缓存项的过期时间 缓存项的依赖关系 - 依赖其它缓存项 缓存项的依赖关系 - 文件依赖 缓存项的移除优先 ...
- 6最好的之一 HTML5/CSS3 演示(PPT)框架
HTML 是全世界最流行的网页编程语言.而HTML5是这门语言的升级版本号.越来越多的开发人员和设计师開始使用HTML5.以下介绍几个 HTML5/CSS3 的演示工具和框架,你能够用它们来创建你的网 ...
- 计算4000000000内的最大f(n)=n值---字符串的问题python实现(五岁以下儿童)
问题: 写一个函数,计算4 000 000 000 以内的最大的那个f(n)=n的值,函数f的功能是统计全部0到n之间全部含有数字1的数字和.比方:f(13)= 6,由于"1"在& ...
- BZOJ 3282 Tree Link-Cut-Tree(LCT)
题目大意: 给定N个点以及每一个点的权值,要你处理接下来的M个操作.操作有4种.操作从0到3编号.点从1到N编号. 0:后接两个整数(x,y),代表询问从x到y的路径上的点的权值的xor和.保证x到y ...