文章出處

回到目錄

關于邏輯刪除

對于邏輯刪除之前的做法是在實體類中加個字段,一般是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這個參數完成邏輯刪除的過濾功能!

是不是很爽,很酷!

對于實體中其它的比較有特點的屬性,而又不是全局的屬性,我們都可以使用接口的方式進行定義,這類似于裝飾模式,即將某個屬性裝飾成某個接口,而在程序的另一端直接去操作這個接口即可

回到目錄


文章列表




Avast logo

Avast 防毒軟體已檢查此封電子郵件的病毒。
www.avast.com


arrow
arrow
    全站熱搜
    創作者介紹
    創作者 大師兄 的頭像
    大師兄

    IT工程師數位筆記本

    大師兄 發表在 痞客邦 留言(0) 人氣()