Entity Framework 6 Recipes 2nd Edition(12-1)译 -> 当SaveChanges( ) 被调用时执行你的代码

时间:2022-10-28 23:13:37

12定制EF

在本章的小节里,定制实体对象和EF处理的一些功能.这些小节将涵盖很多”幕后”的事情,能让你的代码更加统一解决一些事情,比如用一个业务规则中心统一地为实体执行验证.

本章开始的小节,将演示如何在你的应用程序中当调用SaveChanges() 时,执行你自己的代码. 如果你想在你的应用程序里单独定制业务规则,本小节和有些小节非常有用.

在其它小节里,我们将演示如何跟踪数据库连接,如果自动报告集合的改变,如何实现级联删除,如何设定默认值,和如何与强类型的XML属性一起使用.所有这些定制共同点就是扩展对象和EF处理过程使你的代码更具弹性,统一性和可维护性.

12-1. 当SaveChanges( ) 被调用时执行你的代码

问题

你想在data context 中SaveChanges()被调用时,执行你的代码

解决方案

假设你有一个关于申请人的模型(如Figure 12-1所示). 申请人文档做为模型的一部分,你想在Applicant被删除时,文档也一起删除.你可能发现在你应用程序的每个删除Applicant地方,都要删除文档,而你想用一个统一的方式来处理

Entity Framework 6 Recipes 2nd Edition(12-1)译 -> 当SaveChanges( ) 被调用时执行你的代码

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() 方法实现.

附:创建示例用到的数据库的脚本文件