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

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

 

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

 

 

posted @ 2016-01-26 01:19  kid1412  阅读(540)  评论(0编辑  收藏  举报