• 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 Resume ...");

                    var path2 = "JanisRogers.txt";

                    File.AppendAllText(path2, "Janis Rodgers Resume ...");

                    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(" {0} applicants deleted",applicants.Count().ToString());

     

                foreach (var app in applicants)

                {

                    File.Delete(app.ResumePath);

     

                    Console.WriteLine(" {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() 方法实现.

     

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

     

  • 相关阅读:
    java虚拟机-内存的分配
    java-类的多态和多重继承
    java 设计模式-策略模式
    java-线程介绍和基本使用
    java 数据流操作
    java basic
    JAVA连载117-反射的应用与打破封装新
    C连载2-编译过程以及语言历史概览
    Android连载12-完善新闻app内容区域
    JavaScript连载11-Switch
  • 原文地址:https://www.cnblogs.com/kid1412/p/5159175.html
Copyright © 2020-2023  润新知