第12章定制EF
在本章的小节里,定制实体对象和EF处理的一些功能.这些小节将涵盖很多”幕后”的事情,能让你的代码更加统一解决一些事情,比如用一个业务规则中心统一地为实体执行验证.
本章开始的小节,将演示如何在你的应用程序中当调用SaveChanges() 时,执行你自己的代码. 如果你想在你的应用程序里单独定制业务规则,本小节和有些小节非常有用.
在其它小节里,我们将演示如何跟踪数据库连接,如果自动报告集合的改变,如何实现级联删除,如何设定默认值,和如何与强类型的XML属性一起使用.所有这些定制共同点就是扩展对象和EF处理过程使你的代码更具弹性,统一性和可维护性.
12-1. 当SaveChanges( ) 被调用时执行你的代码
问题
你想在data context 中SaveChanges()被调用时,执行你的代码
解决方案
假设你有一个关于申请人的模型(如Figure 12-1所示). 申请人文档做为模型的一部分,你想在Applicant被删除时,文档也一起删除.你可能发现在你应用程序的每个删除Applicant地方,都要删除文档,而你想用一个统一的方式来处理
Listing 12-1. Overriding SaveChanges() to Delete the Resume File When the Applicant Is Deleted
为确保文档随同applicant一起删除, 我们在DbContext 里override SavingChanges() 方法, 我们需要先浏览DbContext里的实体,找出要被删除的Applicant,然后调用真正的SaveChanges() 方法. 最后,为每个被删除的Applicant删除文件,代码如Listing 12-1所示:
Listing 12-1. Overriding SaveChanges() to Delete the Resume File When the Applicant Is Deleted
class Program
{
static void Main(string[] args)
{
using (var context = new EFRecipesEntities())
{
var path1 = "AlexJones.txt";
File.AppendAllText(path1, "Alex Jones\nResume\n...");
var path2 = "JanisRogers.txt";
File.AppendAllText(path2, "Janis Rodgers\nResume\n...");
var app1 = new Applicant
{
Name = "Alex Jones",
ResumePath = path1
};
var app2 = new Applicant
{
Name = "Janis Rogers",
ResumePath = path2
};
context.Applicants.Add(app1);
context.Applicants.Add(app2);
context.SaveChanges();
//删除 Alex Jones
context.Applicants.Remove(app1);
context.SaveChanges();
}
}
}
public partial class EFRecipesEntities
{
public override int SaveChanges()
{
Console.WriteLine("Saving Changes...");
var applicants = this.ChangeTracker.Entries().Where(e => e.State == System.Data.Entity.EntityState.Deleted)
.Select(e => e.Entity).OfType<Applicant>().ToList();
int changes= base.SaveChanges();
Console.WriteLine("\n{0} applicants deleted",applicants.Count().ToString());
foreach (var app in applicants)
{
File.Delete(app.ResumePath);
Console.WriteLine("\n{0}'s resume at {1} deleted",app.Name, app.ResumePath);
}
return changes;
}
}
上述Listing 12-1代码输出结果如下:
Saving Changes...
0 applicants deleted
Saving Changes...
1 applicants deleted
Alex Jones's resume at AlexJones.txt deleted
原理
Listing 12-1里的代码,先插入两个applicant同时创建他们各自的文档.我们的目标是用统一的方式,当applicant被删除时,他们的文档也一同被删除.我们通过override SaveChanges() 方法来实现.在我们的 SaveChanges() 方法中,首先收集所有将被删除的Applicant实例, 接着我们调用真正的
SaveChanges() 方法,把它们从数据库中删除,最后在遍历我们最开始收集的实例,在遍历里删除他们各自的文档. 因为实体从数据库删除后,也会在context中删除,我们不能再通过查询找到context中被删除的裸体,所以第一步里我们用了ToList()方法,先把要被删除的实体保存到另一个对象中.
EF没有实体的插入,更新,修改事件,但是很多想在这些事件里处理的事情,都可以像我们这里的方式一样,通过override SaveChanges() 方法实现.