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)

就很無語。。。。。。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章