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中的任何原生类型,如int
、long
,或者是一个包装类型,如Integer
、Long
。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;
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。