oracle设置mybatis自动生成id插入方式

 更新时间:2023年07月29日 14:17:52   作者:爱丽丝和她的巫师帽  
这篇文章主要介绍了oracle设置mybatis自动生成id插入方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

oracle设置mybatis自动生成id插入

本来想要在插入时自动生成id是很简单的,就是用mybatisPlus的注解,在实体类的id字段上加上@TableId注解,自动生成由雪花算法生成的id,不但随机,而且自增,非常好用

但是现在接手了一个项目没有导入mybaitisPlus,只有mybatis,我不敢乱加东西,只能用mybatis做主键自增了

首先这是一个oracle数据库,和mysql不同,Oracle不能设置自增,只能先创建一个序列(sequence),然后在插入数据时引用该序列的值,

原sql如下:

insert into table_name (ID, name, sex)  
values (sequence_name.NEXTVAL,'张三','男')  

我们从头开始,在创建序列之前,先查询一下已有序列

SELECT * from user_sequences

创建序列

CREATE sequence sequence_name
increment by 1 //以1倍的速度增长,你也可以设置其他数字
start with 1    //从id=1开始增长  
 maxvalue 9999  //最大值为9999,根据情况  
 minvalue 1     //最小值1  
 nocycle        //不循环,也就是一直增长         
 cache 20       //设置缓存cache个序列,如果系统down掉了或者其它情况将会导致序列不连续,也可以设置为---------NOCACHE  
noorder;

mybatis中代码

<insert id="insertAccessLog" parameterType="com.example.AccessLog">
        <selectKey keyProperty="id" resultType="java.lang.Integer" order="BEFORE">
            SELECT YOUR_SEQUENCE_NAME.NEXTVAL FROM DUAL
        </selectKey>
        INSERT INTO your_table_name (ID, name,sex)
        VALUES (#{id}, #{name}, #{sex})
    </insert>

selectKey标签可以在插入语句执行前执行一个查询,以生成主键值。

selectKey标签的常用属性:

  • keyProperty:指定一个Java对象的属性,用于接收生成的主键值。生成的主键值将被设置到该属性中。
  • keyColumn:指定数据库表中的列名,用于接收生成的主键值。生成的主键值将被设置到该列中。
  • resultType:指定生成的主键值的数据类型。可以是Java中的任何原生类型,如intlong,或者是一个包装类型,如IntegerLong
  • order:指定selectKey标签的执行顺序。可以选择在插入语句执行前执行(BEFORE)或执行后执行(AFTER)。默认值是AFTER

注意:java实体类属性的类型和数据库字段类型最好一致,否则在插入时会导致插入时类型不匹配

还有一种更简单的写法

INSERT INTO your_table_name (ID, name,sex)
        VALUES (your_sequence_name, #{name}, #{sex})

下面还有一种情况要注意

mybatis插入一个list,list里是实体类,实体类本身没有生成id,需要oracle数据库使用sequence生成id,如插入List<userDTO>

本来我们使用foreach标签配合values的写法批量插入的,

如下:

insert into PERMANENT_ACCESS_USER (EMPLOYEE_USERACCOUNT,EMPLOYEE_NAME)
            values 
<foreach collection="list" item="item" separator="," close=";">
(#{item.userAccounts},#{item.empName})
</foreach>

或者这样

insert all
        <foreach collection="list" item="item" separator=" ">
            into table_name (name,sex) values ( #{item.name},#{item.sex})
        </foreach>
        SELECT * FROM DUAL

但是现在有个问题,在一个values中只能调用一次NEXTVAL,使得我们没办法在批量插入时给ID赋不同的值

所以现在解决办法是使用触发器,在插入时将序列值赋给id,但是这样的话就需要每张表都要建一个触发器,而触发器又不可控,但是现在又没有找到另外的方法,只能先这样,

具体参考:Oracle sql批量插入多条数据

或者干脆不用序列,直接生成一个随机数,用随机数当id

注意!!!因为用了inser into。。。select的语法,所以useGeneratedKeys="false"这个一定要加在insert标签里面,否则会报错

这个坑踩了好久,一度误以为mybatis不支持inser into。。。select语法,一定要注意

useGeneratedKeys="false"
<insert id="insertPermanentUser" parameterType="list" useGeneratedKeys="false">
        INSERT INTO PERMANENT_ACCESS_USER(id, name, sex)
        SELECT id, name, sex FROM
        <foreach collection="list" item="item" separator=" union all " open="(" close=")">
            SELECT dbms_random.VALUE(0,10000000) as id,#{item.name} as name,#{item.sex} as sex from dual
        </foreach>
    </insert>

oracle中自动生成id的函数以及注意事项

Oracle中自动生成id的函数 :sys_guid()

SELECT sys_guid() FROM aTable a ;

注意:上面这个是可以自动生成id,但是很多时候自动生成的id会出现乱码

原因:SYS_GUID 以16位RAW类型值形式返回一个全局唯一的标识符,而我们一般用的是十六进制的字符

所有用以下的解决方式:

-- 用大写或者小写函数都能解决 : 因为大小写会把raw串自动转换为十六进制
select lower(sys_guid()) from aTable a;
-- 或者 
select upper(sys_guid()) from aTable a;
-- 个人推荐这种:因为它不把值转换成大小写
-- rawtohex函数: 将raw串转换为十六进制
select rawtohex(sys_guid()) from aTable a;
-- 还有
select lower(rawtohex(sys_guid())) from aTable a;

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • [Oracle] 如何使用触发器实现IP限制用户登录

    [Oracle] 如何使用触发器实现IP限制用户登录

    在Oracle里,不像MySQL那样方便,可以直接在用户上进行IP限制,Oracle要实现用户级别的IP限制,可以使用触发器来迂回实现,以下就是示例,需要的朋友可以参考下
    2013-07-07
  • delete archivelog all无法清除归档日志解决方法

    delete archivelog all无法清除归档日志解决方法

    最近在因归档日志暴增,使用delete archivelog all貌似无法清除所有的归档日志,究竟是什么原因呢?本文将为您解答,需要的朋友可以参考下
    2012-12-12
  • Oracle的CLOB大数据字段类型操作方法

    Oracle的CLOB大数据字段类型操作方法

    VARCHAR2既分PL/SQL Data Types中的变量类型,也分Oracle Database中的字段类型,不同场景的最大长度不同。接下来通过本文给大家分享Oracle的CLOB大数据字段类型操作方法,感兴趣的朋友一起看看吧
    2017-08-08
  • Oracle数据更改后出错的解决方法

    Oracle数据更改后出错的解决方法

    这篇文章主要介绍了Oracle数据更改后出错的解决方法,需要的朋友可以参考下
    2014-07-07
  • Oracle中Like与Instr模糊查询性能大比拼

    Oracle中Like与Instr模糊查询性能大比拼

    本文通过实例代码给大家介绍了Oracle中Like与Instr模糊查询性能对比,需要的朋友参考下吧
    2017-05-05
  • Oracle 监听器密码设置方法(LISTENER)

    Oracle 监听器密码设置方法(LISTENER)

    在缺省的情况下,任意用户不需要使用任何密码即通过lsnrctl 工具对Oracle Listener进行操作或关闭,从而造成任意新的会话都将无法建立连接
    2016-09-09
  • Oracle数据库中通用的函数实例详解

    Oracle数据库中通用的函数实例详解

    OracleSQL提供了用于执行特定操作的专用函数,这些函数大大增强了SQL语言的功能,下面这篇文章主要给大家介绍了关于Oracle数据库中通用函数的相关资料,需要的朋友可以参考下
    2022-03-03
  • Oracle如何清除一个用户下的所有表(谨慎操作!)

    Oracle如何清除一个用户下的所有表(谨慎操作!)

    在测试数据库脚本可用性的时候,会新建一个用户然后执行脚本,测试成功之后,需要清空表,下面这篇文章主要给大家介绍了关于Oracle如何清除一个用户下的所有表的相关资料,需要的朋友可以参考下
    2023-03-03
  • oracle学习笔记(三)

    oracle学习笔记(三)

    最近需要用的oracle,所以大家好好的学习下基础并整理下资料,希望能帮助到需要的朋友。
    2011-12-12
  • oracle 日期时间函数使用总结

    oracle 日期时间函数使用总结

    经常写 sql 的同学应该会接触到一些 oracle 的日期时间函数, 例如: 财务软件或者人力资源软件需要按照每年, 每季度, 每月, 甚至每个星期来进行统计
    2014-05-05

最新评论