通用业务设计---如何解决'逻辑删除'和'唯一索引'冲突的问题

如何解决'逻辑删除'和'唯一索引'冲突的问题

概括

简单一句话

把del_flag 0值:代表未删除 null值:代表删除

场景

当要对保证某些可编辑字段的唯一性时,代码需要在插入和更新时都进行唯一性校验,这很繁琐。

因此想到采用将对于唯一性的校验直接交给数据库进行,通过数据库唯一索引实现。

但这就又出现了另一个问题:由于采用的是"逻辑删除",那么会造成唯一字段值相同的数据只能删除一次的情况。第二次删除时,由于已存在 del_flag 值为 1 的数据,会报错"违反唯一约束"

解决方案

基于mysql"唯一约束对 Null 失效"的原理,将被删数据的逻辑删除字段置为 null 即可。如果使用的是 mybatis-plus,可以直接通过@TableLogic注解实现

java 复制代码
    /**
     * 是否删除
     *
     * <p>
     *     为解决'逻辑删除'和'唯一索引'冲突问题,而将逻辑删除字段设置为NULL
     * </p>
     */
    @TableLogic(value = "0", delval = "NULL")
    private Boolean deleteFlag;

这样就可以在数据库层面轻松实现唯一性的保证,且'逻辑删除'和'唯一索引'不会冲突。

此时当入参违反唯一性约束时就会抛出异常,我们需要对异常进行处理,以向前端返回更加友好的报错信息。

java 复制代码
try {
    thisGiftProductService.saveThisGiftProduct(thisGiftProductDO, request);
    return RestResult.<Void>builder().success("保存成功");
} catch (Exception e) {
    log.error("新增错误", e);
    Throwable cause = e.getCause();
    if (cause instanceof java.sql.SQLIntegrityConstraintViolationException) {
        return RestResult.<Void>builder().fail("本品产品编码和组织信息组合不能重复");
    }
    return RestResult.<Void>builder().fail(e.getMessage());
}

核心在于:

java 复制代码
log.error("新增错误", e);
Throwable cause = e.getCause();
if (cause instanceof java.sql.SQLIntegrityConstraintViolationException) {
    return RestResult.<Void>builder().fail("本品产品编码和组织信息组合不能重复");
}

之所以采用这种方法,是因为 java.sql.SQLIntegrityConstraintViolationException 异常是无法作为 catch 参数捕捉的。

相关推荐
陈橘又青15 分钟前
100% AI 写的开源项目三周多已获得 800 star 了
人工智能·后端·ai·restful·数据
2501_9411486141 分钟前
C++实时数据处理实战:多线程与异步IO结合高性能代码解析
java·后端·struts
IT_陈寒1 小时前
Redis实战:5个高频应用场景下的性能优化技巧,让你的QPS提升50%
前端·人工智能·后端
mzlogin1 小时前
借助 Let's Encrypt 节省 SSL 证书费用
后端·devops
虎子_layor1 小时前
单机压测从百到三千:一次短链跳转服务的全链路性能优化实战
后端·性能优化
SelectDB1 小时前
Apache Doris 中的 Data Trait:性能提速 2 倍的秘密武器
数据库·后端·apache
zhengzizhe2 小时前
LangGraph4j LangChain4j JAVA 多Agent编排详解
java·后端
程序员鱼皮2 小时前
又被 Cursor 烧了 1 万块,我麻了。。。
前端·后端·ai·程序员·大模型·编程
福大大架构师每日一题2 小时前
2025-11-27:为视频标题生成标签。用go语言,给定一个字符串 caption(视频标题),按下面顺序处理并输出一个标签: 1. 将标题中的各个词合并成一
后端
程序员爱钓鱼2 小时前
Go语言 OCR 常用识别库与实战指南
后端·go·trae