回到目录

关于逻辑删除

对于逻辑删除之前的做法是在实体类中加个字段,一般是status,其中一种状态是删除,当然也有其它做法,如加个bool的字段IsDeleted,这些其实都过于武断,即它在基类里加上后,所以实体类都会有这种特性,而对于现实的数据表,可能不显示这种逻辑删除的特性,如关系表,日志表,可能删除就是物理上的直接delete,而这种删除字段加上去,我们的做也是在业务层手动调用update方法,或者在底层提供一个delete方法的重载,总之,感觉不是很爽!

看了ABP的软删除之后,对大叔有了新的启发,即提出一个逻辑删除的接口,所以需要这个字段的实体都去实现这个接口即可!

逻辑删除的接口(对实体属性的装饰)

   /// <summary>
/// 具有逻辑删除的接口,实体需要实现这个接口,将IsDeleted实现
/// 在仓储实现类中,delete方法判断实体是否实现了ILogicDeleteBehavor这个接口,然后再决定是否逻辑删除
/// </summary>
public interface ILogicDeleteBehavor
{
/// <summary>
/// 是否已经删除,默认为false
/// </summary>
bool IsDeleted { get; set; }
}

这个接口很干净,只有一个属性,这个属性用来标识删除的状态,true表示已经删除,在进行select操作时我们需要将这个状态过滤,在delete方法里,我们也可以通过判断当前实体是否属于ILogicDeleteBehavor接口而对它采取是否进行逻辑删除!

实体多继承一个接口,完成某个特定的功能

 public partial class WebManageUsers :
Lind.DDD.Domain.Entity,
ILogicDeleteBehavor,
IStatusBehavor
{
#region IStatusBehavor 成员 public Status DataStatus { get; set; } #endregion #region ILogicDeleteBehavor 成员 public bool IsDeleted { get; set; } #endregion
public WebManageUsers()
{
this.WebManageRoles = new HashSet<WebManageRoles>();
this.WebDepartments = new HashSet<WebDepartments>();
}
[DisplayName("登陆名"), Required]
public string LoginName { get; set; }
[DisplayName("密码"), Required]
public string Password { get; set; }
[DisplayName("真实姓名"), Required]
public string RealName { get; set; }
[DisplayName("手机")]
public string Mobile { get; set; }
[DisplayName("电子邮件")]
public string Email { get; set; }
[DisplayName("描述")]
public string Description { get; set; }
[DisplayName("操作者")]
public string Operator { get; set; }
[DisplayName("所属项目")]
public Nullable<int> WebSystemID { get; set; }
public virtual ICollection<WebManageRoles> WebManageRoles { get; set; }
public virtual ICollection<WebDepartments> WebDepartments { get; set; } }

本例采用的是EF的CodeFirst方式,所以需要将自己定义实体,然后根据实体自动生成数据库.

删除方法直接判断实体是否实现了某个接口

 public void Delete(TEntity item)
{
if (item != null)
{
if (item is ILogicDeleteBehavor)
{
//逻辑删除
var pkList = GetPrimaryKey().Select(i => i.Name);
var entityType = typeof(TEntity);
List<object> primaryArr = new List<object>();
foreach (var primaryField in pkList)
{
primaryArr.Add(entityType.GetProperty(primaryField).GetValue(item, null));
}
var old = this.Find(primaryArr.ToArray());
(old as ILogicDeleteBehavor).IsDeleted = true;
this.Update(old);
}
else
{
//物理删除
Db.Set<TEntity>().Attach(item as TEntity);
Db.Entry(item).State = EntityState.Deleted;
Db.Set<TEntity>().Remove(item as TEntity);
this.SaveChanges();
} }
}

上面的设置,对于在列表里删除某个对象已经可以实现了,而如何去过滤列表里的记录呢,当然直接在DbSet<TEntity>()里去过滤是最好的,但没有直接的方式,因为我们的IsDeleted属性没有对外暴露,而对于Linq to Entity来说,你无法在查询表达式中输

入EDM之外的元素名(那也不认,它只认实体类型),还好在ABP里我找到了不错的方法,就是在数据上下文的OnModelCreating方法上,添加过滤器,这个过滤器需要我们安装EntityFramework.DynamicFilters包,直接用Nuget可以安装.

ModelBuilder.Filter完成对集合的全局过滤

  protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
modelBuilder.Filter("LogicDelete", (Lind.DDD.Domain.ILogicDeleteBehavor d) => d.IsDeleted, false);
}

这样在所有Linq Select语句之前都会添加d.IsDeleted==false这个参数完成逻辑删除的过滤功能!

是不是很爽,很酷!

对于实体中其它的比较有特点的属性,而又不是全局的属性,我们都可以使用接口的方式进行定义,这类似于装饰模式,即将某个属性装饰成某个接口,而在程序的另一端直接去操作这个接口即可

回到目录

Lind.DDD.ILogicDeleteBehavor~逻辑删除的实现的更多相关文章

  1. Lind.DDD敏捷领域驱动框架~介绍

    回到占占推荐博客索引 最近觉得自己的框架过于复杂,在实现开发使用中有些不爽,自己的朋友们也经常和我说,框架太麻烦了,要引用的类库太多:之前架构之所以这样设计,完全出于对职责分离和代码附复用的考虑,主要 ...

  2. Lind.DDD.Caching分布式数据集缓存介绍

    回到目录 戏说当年 大叔原创的分布式数据集缓存在之前的企业级框架里介绍过,大家可以关注<我心中的核心组件(可插拔的AOP)~第二回 缓存拦截器>,而今天主要对Lind.DDD.Cachin ...

  3. Lind.DDD.API核心技术分享

    回到目录 关于Lind.DDD框架里API框架的技术点说明 讲解:张占岭 花名:仓储大叔 主要框架:Lind.DDD 目录 关于Lind.DDD.Authorization 关于授权的原理 关于Api ...

  4. Lind.DDD.LindAspects方法拦截的介绍

    回到目录 什么是LindAspects 之前写了关于Aspects的文章<Lind.DDD.Aspects通过Plugins实现方法的动态拦截~Lind里的AOP>,今天主要在设计思想上进 ...

  5. Lind.DDD.Aspects通过Plugins实现方法的动态拦截~Lind里的AOP

    回到目录 .Net MVC之所以发展的如些之好,一个很重要原因就是它公开了一组AOP的过滤器,即使用这些过滤器可以方便的拦截controller里的action,并注入我们自己的代码逻辑,向全局的异常 ...

  6. Lind.DDD.Repositories.Mongo层介绍

    回到目录 之前已经发生了 大叔之前讲过被仓储化了的Mongodb,而在大叔开发了Lind.DDD之后,决定把这个东西再搬到本框架的仓储层来,这也是大势所趋的,毕竟mongodb是最像关系数据库的NoS ...

  7. Lind.DDD.Repositories.Redis层介绍

    回到目录 之前已经发生了 大叔之前介绍过关于redis的文章,有缓存,队列,分布式pub/sub,数据集缓存以及仓储redis的实现等等,而今天在Lind.DDD的持久化组件里,redis当然也有一席 ...

  8. Lind.DDD.Events领域事件介绍

    回到目录 闲话多说 领域事件大叔感觉是最不好讲的一篇文章,所以拖欠了很久,但最终还是在2015年年前(阴历)把这个知识点讲一下,事件这个东西早在C#1.0时代就有了,那时学起来也是一个费劲,什么是委托 ...

  9. Lind.DDD.Authorization用户授权介绍

    回到目录 Lind.DDD.Authorization是Lind.DDD框架的组成部分,之所以把它封装到框架里,原因就是它的通用性,几乎在任何一个系统中,都少不了用户授权功能,用户授权对于任何一个系统 ...

随机推荐

  1. HTML中em与b等的区别

    最近在学习HTML基本知识,看到b标签和strong还有em,都是表示强调目的的,那他们之间的区别是什么呢?总结如下: < b > < i > 是视觉要素(presentati ...

  2. avascript中的this与函数讲解

    徐某某 一个半路出家的野生程序员 javascript中的this与函数讲解 前言 javascript中没有块级作用域(es6以前),javascript中作用域分为函数作用域和全局作用域.并且,大 ...

  3. css双飞翼布局

     双飞翼布局是一种比较灵活的布局,始于淘宝UED,玉伯提出的,他着重介绍的是双飞翼栅格布局. 三列布局为"双飞燕"布局,可以把三栏比作一只鸟,main部分相当是于鸟的身体,而lef ...

  4. maven的聚合与继承

    新建一个空的maven项目user-parent Pom.xml内容 <project xmlns="http://maven.apache.org/POM/4.0.0" x ...

  5. Ajax_01之概述、响应

    1.URL.URI和URN URL:Unified Resource Locator:统一资源定位符: URI:Unified Resource Identifier:统一资源识别符: URN:Uni ...

  6. iOS 真机调试不能连接网络的排错过程

    开发环境: macOS 10.12.1 Xcode 8.1 Qt 5.8 gSOAP 2.8 iPhone 6S+iOS 10.1.1   问题: 使用 Qt Quick 写了一个跨平台的应用,在Wi ...

  7. Constraint3:check约束 和 null

    Check约束用以限制单列或多列的可能取值范围. 1,在check约束中(check(expression)),如果expression返回的结果是Unknown,那么check返回的结果是true. ...

  8. IOS 开发中 Whose view is not in the window hierarchy 错误的解决办法

    在 IOS 开发当中经常碰到 whose view is not in the window hierarchy 的错误,该错误简单的说,是由于 "ViewController" ...

  9. 轻量级前端MVVM框架avalon - 模型转换

    接上一章 ViewModel modelFactory工厂是如何加工用户定义的VM? 附源码 洋洋洒洒100多行内部是魔幻般的实现 1: function modelFactory(scope) { ...

  10. 配置 L2 Population - 每天5分钟玩转 OpenStack(114)

    前面我们学习了L2 Population 的原理,今天讨论如何在 Neutron 中配置和启用此特性. 目前 L2 Population 支持 VXLAN with Linux bridge 和 VX ...