4

我在我的 MVC 应用程序中使用 EF 5 Database first 方法。我所有的表都使用一个名为 Deleted 的字段,它是一个布尔字段,用于标记一条记录已被删除。

我试图摆脱每次查询数据库时都必须检查 Deleted == false 的要求。执行此操作的非常简单的方法是在 edmx 文件中使用条件映射,其中 EF 始终返回未删除的数据。这都很好。

但是做这个条件映射的问题是,当我想允许用户从他们的地址簿中删除一些记录时,我无权访问 EF 中的 Delete 字段,因为我在条件映射中使用了它,因此我必须寻找另一个选项来允许用户删除记录。

我想的方式是创建一个存储过程来处理删除查询并在我想删除记录时调用它。

有没有更好的方法来做到这一点?即使在条件映射中使用删除字段,是否也可以访问它?

4

1 回答 1

14

在实体框架代码优先中为软删除提供了一个可行的解决方案,这可能会有所帮助。

关键是您为每个希望能够软删除的模型添加一个鉴别器。在代码中首先是这样完成的:

modelBuilder.Entity<Foo>().Map(m => m.Requires("IsDeleted").HasValue(false));

这使得它对上下文不可见,因此您必须使用 sql 进行删除。

如果这相当于 Database First 中的“条件映射”,那么修改 sql 的一种方法是覆盖 SaveChanges 并从那里运行 sql:

public override int SaveChanges()
{
   foreach (var entry in ChangeTracker.Entries()
             .Where(p => p.State == EntityState.Deleted 
             && p.Entity is ModelBase))//I do have a base class for entities with a single 
                                       //"ID" property - all my entities derive from this, 
                                       //but you could use ISoftDelete here
    SoftDelete(entry);

    return base.SaveChanges();
}

private void SoftDelete(DbEntityEntry entry)
{
    var e = entry.Entity as ModelBase;
    string tableName = GetTableName(e.GetType());
    Database.ExecuteSqlCommand(
             String.Format("UPDATE {0} SET IsDeleted = 1 WHERE ID = @id", tableName)
             , new SqlParameter("id", e.ID));

    //Marking it Unchanged prevents the hard delete
    //entry.State = EntityState.Unchanged;
    //So does setting it to Detached:
    //And that is what EF does when it deletes an item
    //http://msdn.microsoft.com/en-us/data/jj592676.aspx
    entry.State = EntityState.Detached;
}

用于获取表名的方法解释here

这就是我过去的做法。可能与您在 EF5 中的数据库优先方法无关,但我现在已转向在存储过程中执行此操作。EF6 Code FirstCreateStoredProcedure在迁移文件中生成调用。我将这些替换为this.CreateDeleteProcedure("dbo.Foo_Delete", "[dbo].[Foos]");- 这是对我自己的扩展方法的调用:

public static class MigrationExtensions
{
    internal static string DeleteSqlFormat
    {
        //I also hard delete anything deleted more than a day ago in the same table
        get { return "DELETE FROM {0} WHERE IsDeleted = 1 AND DATEADD(DAY, 1, DeletedAt) < GETUTCDATE(); UPDATE {0} SET IsDeleted = 1, DeletedAt = GETUTCDATE() WHERE ID = @ID;"; }
    }

    internal static void CreateDeleteProcedure(this DbMigration migration, string procName, string tableName)
    {
        migration.CreateStoredProcedure(
                        procName,
                        p => new
                        {
                            ID = p.Int(),
                        },
                        body:

                            string.Format(MigrationExtensions.DeleteSqlFormat, tableName)

                    );
    }

}
于 2013-10-08T12:50:37.187 回答