当前位置: 首页 > news >正文

springboot集成达梦数据库及SET IDENTITY_INSERT为ON时问题

集成

pom.xml

<!--   达梦数据库     -->
<dependency>
    <groupId>com.dameng</groupId>
    <artifactId>Dm8JdbcDriver18</artifactId>
    <version>8.1.1.49</version>
</dependency>

<dependency>
    <groupId>com.dameng</groupId>
    <artifactId>DmDialect-for-hibernate5.0</artifactId>
    <version>8.1.1.49</version>
</dependency>

<dependency>
    <!--注意:只有这个版本的hibernate兼容达梦数据库 -->
    <groupId>org.hibernate</groupId>
    <artifactId>hibernate-core</artifactId>
    <version>5.3.18.Final</version>
</dependency>

application.yml

spring:
  datasource:
    url: jdbc:dm://127.0.0.1:5236?schema=xxxxx
    username: SYSDBA
    password: 123456789
    type: com.alibaba.druid.pool.DruidDataSource
    driverClassName: dm.jdbc.driver.DmDriver
    druid: # 记得将merge-sql相关关闭
  jpa:
    properties:
      hibernate:
        dialect: org.hibernate.dialect.DmDialect

Entity#id说明

@Id
@GeneratedValue(strategy = GenerationType.SEQUENCE)
private Integer id = 0;

# IDENTITY:mysql有效
# SEQUENCE:达梦有效

达梦配置 - 基础操作mapper

public interface DmSQLMapper {

    void on(@Param("tableName") String tableName);


    void off(@Param("tableName") String tableName);

}

// 对应的xml 
<update id="on">
    set IDENTITY_INSERT ${tableName} ON;
</update>

<update id="off">
    set IDENTITY_INSERT ${tableName} OFF;
</update>

达梦配置 - 自定义保存注解

@Documented
@Target({ElementType.PARAMETER,
        ElementType.METHOD,
        ElementType.TYPE_USE,
        ElementType.TYPE_PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
public @interface DmSave {

    String[] className() default "";
}

达梦配置 - 保存监听

@Component
@Slf4j
public class DmListener {

    @Autowired
    private DmSQLMapper dmSQLMapper;


    /**
     * 在保存之前调用
     */
    @PrePersist
    public void prePersist(Object source){
        String name = source.getClass().getAnnotation(Table.class).name();

        log.info("表名: " + name);
        dmSQLMapper.on(name);
    }

    /**
     * 在保存之后调用
     */
    @PostPersist
    public void postPersist(Object source){
        String name = source.getClass().getAnnotation(Table.class).name();
        dmSQLMapper.off(name);
        log.info("表名: " + name);
    }

}

达梦配置 - 日志切面

@Aspect
@Component
@Slf4j
public class DmSystemLogAspect {

    @Autowired
    private DmSQLMapper dmSQLMapper;


    @Pointcut("@annotation(xxx.annotataion.DmSave)")
    public void controllerAspect(){

    }

    @Before("controllerAspect()")
    public void on(JoinPoint point){
        // 获取注解中的参数值
        MethodSignature methodSignature = (MethodSignature)point.getSignature();
        Method method = methodSignature.getMethod();

        // 获取注解Action
        DmSave annotation = method.getAnnotation(DmSave.class);

        // 获取了参数类名
        String[] strings = annotation.className();

        if (strings.length > 0){
            // 进行达梦数据库on操作
            for (String tableName : strings) {
                log.info("开启类名" + tableName);

                dmSQLMapper.on(tableName);
            }
        }
    }
}

出现的问题

仅当指定列列表,且SET IDENTITY_INSERT为ON时

说明

默认情况下,达梦数据库是不允许对自增的列(例如:id)进行插入操作的。如果使用mybatis手动sql进行insert,应该不会出现很大的问题。但是,使用 jpa 进行save或者saveAll的都是实体,默认是携带id字段的(虽然为null或者0),但是还是报错。

解决

在保存之前,使用语句 set IDENTITY_INSERT ${tableName} ON 即可解决问题,就像上面的mapper文件一样,每次save或者saveAll之前手动调用一次,或者使用监听器或者切面统一进行处理

在同一个数据库连接中,同时只能针对一张表进行该操作的开启,如果开启下一张表,那么上一张会自动关闭

另外一种情况

如果使用了该语句,还没解决问题,注意是否在方法或者类上加了事务注解(@Transactional)
例如:

@Transactional(rollbackFor = Exception.class)
public void saveData(){
	dmSQLMapper.on("user");
	userRepository.save(user);
	
	dmSQLMapper.on("user_role");
	userRoleRepository.saveAll(userRoleList);
}

在上面这个案例中,虽然save或saveAll之前,都对相应的表进行了开启,但是执行时,还是会报出 SET IDENTITY_INSERT 的错误。
这是因为在开启了事务之后,sql语句的执行顺序发生了变化,真正需要保存数据的sql放到了对数据不产生影响的sql的后面,而且这种错误很难发现,在控制台是打印不出来的,只有在开启了SQL打印的情况下,跟踪每一条sql才会发现。

认为SQL执行的顺序:

  1. set IDENTITY_INSERT user ON
  2. insert into user(id, name) values(0, “zhangsan”)
  3. set IDENTITY_INSERT user_role ON
  4. insert into user_role(id, user_id, role_id) values(0, 1, 1)

实际SQL执行顺序:

  1. set IDENTITY_INSERT user ON
  2. set IDENTITY_INSERT user_role ON
  3. insert into user(id, name) values(0, “zhangsan”) # 在此处就报错了
  4. insert into user_role(id, user_id, role_id) values(0, 1, 1)

就很无语。。。。。。

在这里插入图片描述

相关文章:

  • 在 Python 中,函数是第一类对象
  • Hadoop大数据实战笔记
  • 彻底理解Java并发:Java内存模型
  • Ubuntu22.04如何开机重新自动运行脚本
  • USB应用实战视频教程第3期:手把手玩转USB BULK方式下位机和QT6.4上位机开发(上篇)
  • Windows环境下增加c/c++编译堆栈空间的方法(CMD,DEV,CLion,VS2022)
  • Vue Transition 过渡组件 + animate.style 动画库的使用
  • 【网络工程师笔记】——ACL
  • java计算机毕业设计学生生活管理源码+系统+数据库+lw文档
  • python环境搭建
  • 基于simulink的超级电容,电池及DC motor充放电系统仿真
  • JS(DOM)第十五课
  • Java多线程(2)
  • 深入浅出Spring注解(22)
  • [C++基础]-初识模板
  • [js高手之路]搞清楚面向对象,必须要理解对象在创建过程中的内存表示
  • [译]前端离线指南(上)
  • 【从零开始安装kubernetes-1.7.3】2.flannel、docker以及Harbor的配置以及作用
  • 002-读书笔记-JavaScript高级程序设计 在HTML中使用JavaScript
  • css系列之关于字体的事
  • interface和setter,getter
  • Java 实战开发之spring、logback配置及chrome开发神器(六)
  • javascript从右向左截取指定位数字符的3种方法
  • Lucene解析 - 基本概念
  • SpringBoot 实战 (三) | 配置文件详解
  • Tornado学习笔记(1)
  • 工作踩坑系列——https访问遇到“已阻止载入混合活动内容”
  • 盘点那些不知名却常用的 Git 操作
  • 批量截取pdf文件
  • 普通函数和构造函数的区别
  • 前端攻城师
  • 强力优化Rancher k8s中国区的使用体验
  • 入口文件开始,分析Vue源码实现
  • 我的业余项目总结
  • 我与Jetbrains的这些年
  • #define用法
  • #多叉树深度遍历_结合深度学习的视频编码方法--帧内预测
  • #我与Java虚拟机的故事#连载09:面试大厂逃不过的JVM
  • $redis-setphp_redis Set命令,php操作Redis Set函数介绍
  • (12)Hive调优——count distinct去重优化
  • (6)设计一个TimeMap
  • (附源码)计算机毕业设计SSM智慧停车系统
  • (一)spring cloud微服务分布式云架构 - Spring Cloud简介
  • 、写入Shellcode到注册表上线
  • ..thread“main“ com.fasterxml.jackson.databind.JsonMappingException: Jackson version is too old 2.3.1
  • .net wcf memory gates checking failed
  • .NET 表达式计算:Expression Evaluator
  • .NET 同步与异步 之 原子操作和自旋锁(Interlocked、SpinLock)(九)
  • .net 托管代码与非托管代码
  • .Net程序猿乐Android发展---(10)框架布局FrameLayout
  • .Net调用Java编写的WebServices返回值为Null的解决方法(SoapUI工具测试有返回值)
  • .pop ----remove 删除
  • // an array of int
  • @Valid和@NotNull字段校验使用
  • [ 2222 ]http://e.eqxiu.com/s/wJMf15Ku