zoukankan      html  css  js  c++  java
  • Jpa实现逻辑删除

    Spring Data 是个好东西,极大简化了后端dao的操作,只需要在 dao 接口写个 findByXXX 的方法就能自动实现按条件查询这个简直太爽了。但是在实际使用过程中,可能会遇到一个持久化逻辑删除的问题。那么问题来了。spring data jpa并不支持逻辑删除。那如何处理?

    在互联网项目中,通常删除都不是物理删除,而是逻辑删除

    那么在展示数据的时候需要过滤掉已删除的数据。而@Where 注解可以说就是为此而设计的。

    package org.hibernate.annotations;
    
    import java.lang.annotation.Retention;
    import java.lang.annotation.Target;
    
    import static java.lang.annotation.ElementType.FIELD;
    import static java.lang.annotation.ElementType.METHOD;
    import static java.lang.annotation.ElementType.TYPE;
    import static java.lang.annotation.RetentionPolicy.RUNTIME;
    
    /**
     * Where clause to add to the element Entity or target entity of a collection.  The clause is written in SQL.
     * A common use case here is for soft-deletes.
     *
     * @author Emmanuel Bernard
     */
    @Target({TYPE, METHOD, FIELD})
    @Retention(RUNTIME)
    public @interface Where {
        /**
         * The where-clause predicate.
         */
        String clause();
    }

     这玩意儿乍一看很美好。但是实际不然。这样做将所有的删除方法都覆盖了。当我想要用到物理删除的时候。就不能用了。于是再次出发。寻找一种更加美妙的解决方案。终于让我找到了。话不多说。直接放码。

    @NoRepositoryBean
    public interface BaseDao<T extends BaseEntry, ID extends Serializable> extends PagingAndSortingRepository<T, ID> {
     
        @Override
        @Transactional(readOnly = true)
        @Query("select e from #{#entityName} e where e.deleted = false")
        List<T> findAll();
     
        @Override
        @Transactional(readOnly = true)
        @Query("select e from #{#entityName} e where e.id in ?1 and e.deleted = false")
        Iterable<T> findAll(Iterable<ID> ids);
     
        @Override
        @Transactional(readOnly = true)
        @Query("select e from #{#entityName} e where e.id = ?1 and e.deleted = false")
        T findOne(ID id);
     
        @Override
        @Transactional(readOnly = true)
        @Query("select count(e) from #{#entityName} e where e.deleted = false")
        long count();
     
        @Override
        @Transactional(readOnly = true)
        default boolean exists(ID id) {
            return findOne(id) != null;
        }
     
        @Query("update #{#entityName} e set e.deleted = true where e.id = ?1")
        @Transactional
        @Modifying
        void logicDelete(ID id);
     
        @Transactional
        default void logicDelete(T entity) {
            logicDelete((ID) entity.getId());
        }
     
        @Transactional
        default void logicDelete(Iterable<? extends T> entities) {
            entities.forEach(entity -> logicDelete((ID) entity.getId()));
        }
     
        @Query("update #{#entityName} e set e.deleted = true ")
        @Transactional
        @Modifying
        void logicDeleteAll();
    }

    BaseEntry代码如下:

        @Data
        @MappedSuperclass
        public class BaseEntry implements Serializable {
         
            private static final long serialVersionUID = 5966306766659220492L;
         
            @Id
            protected String id;
         
            @Temporal(TemporalType.TIMESTAMP)
            protected Date createdDate;
         
            protected String createdBy;
         
            @Temporal(TemporalType.TIMESTAMP)
            protected Date updatedDate;
         
            protected String updatedBy;
         
            protected Boolean deleted = false;
        }
  • 相关阅读:
    遍历数据类型数组方式
    for 循环 和for..in循环遍历数组 的区别
    多个区域内有相同属性名称子元素,同一区域内 操作DOM子集 使用$("选择器",context)方法
    CSS 使用技巧
    JavaScript 动态加载页面 js文件
    angular2环境配置
    在路上●我的年青●逐步前进
    ARM v8-A 系列CPU的MMU隐射分析
    ARM Cortex-A53 Cache与内存的映射关系以及Cache的一致性分析
    二维图像的投影和图像重建分析之傅里叶变换法
  • 原文地址:https://www.cnblogs.com/zouhong/p/14365880.html
Copyright © 2011-2022 走看看