Mybatis的selectKey和useGeneratedKeys区别

https://blog.csdn.net/u011118321/article/details/66973634

useGeneratedKeys 要求数据库本身具备主键自动增长的功能,比如说,mysql,sqlserver可以使用useGeneratedKeys =true 这功能,不支持主键自动增长的数据库是不能使用useGeneratedKeys =true的

不支持的用selectKey子节点来做,oracle不支持useGeneratedKeys

否则会报
“不支持的特性”

http://xiaoyue26.github.io/2015/10/04/mybatis%E5%A4%84%E7%90%86mysql%E8%87%AA%E5%A2%9E%E4%B8%BB%E9%94%AE%E7%9A%84%E4%B8%89%E7%A7%8D%E5%86%99%E6%B3%95/

mybatis处理mysql自增主键的三种写法

  1. 数据库的自增主键id(int(11)),查看auto_increment,记录的是下一次插入时的主键值。(=最大的id+1)

    1
    2
    3
    
    SHOW TABLE STATUS;
    #或:
    SELECT AUTO_INCREMENT FROM  information_schema.tables WHERE table_name='t_account_0'
    
  2. 测试select @@identity命令:

    1
    2
    
    INSERT into user_info  (uname, unumber) VALUES ('test','test');
    select @@identity  as 'curMaxId';
    

注意到第一条命令并没有插入id值,以便让数据库插入自增id值。
两条命令同一线程中顺序执行后,第二条命令获取到第一条命令插入的id值。
例如,若select @@identity结果为106,则auto_increment此时为107,而INSERT的记录id值为106.

  1. 第一种mybatis支持自增主键的写法,使用useGeneratedKeyskeyProperty属性设置:
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    
    <sql id="UserInfoSet">
    		<set>
    			<if test="id != null">id=#{id},</if>
    			<if test="uname!= null">uname=#{uname},</if>
    			<if test="unumber != null">unumber=#{unumber},</if>
    		</set>
    	</sql>
    <insert id="insert" parameterType="UserInfo"
    		 useGeneratedKeys="true" keyProperty="id">
    		insert user_info
    		<include refid="UserInfoSet" />
    </insert>
    

如果表中主键已经设定为auto_increment,则插入后会自动把生成的id值返回给插入的对象。

1
2
3
4
5
6
UserInfo userInfo = new UserInfo();
userInfo.setUname("xiaoming"); 
userInfo.setUnumber("x07");
//a.插入前userinfo的id为null
result = userService.insert(userInfo);		
//b.插入后userinfo的id为数据库生成的id值(例如123).

 

  1. 第二种使用selectKey,限定于mysql数据库的写法,其他数据库如oraclebefore,各有不同:
    1
    2
    3
    4
    5
    6
    
    <insert id="insert" parameterType="UserInfo">
    	<selectKey resultType="java.lang.Integer" keyProperty="id" order="AFTER">
    	SELECT @@IDENTITY
        </selectKey>
    insert user_info <include refid="UserInfoSet" />
    </insert>
    

此时表现和上一种一样;
若把SELECT @@IDENTITY改为 select 20 from dual , 则数据库中虽依然插入了正常递增的id值(如123),
b处插入后的对象中id值为20. 可见selectKey标签实际执行是在insert执行后,返回obj对象前,将其中的id值改为20.并不影响insert执行过程。
综上所示,这两种方法其实都不影响insert,只是执行后获取一次id值。insert过程中生成id值是由数据库支持的。
所以如果数据库中没有设定主键的自增属性,这两种都是无能为力的,此时就要更改SelectKey的用法:

  1. 第三种,使用select max(id)+1
    1
    2
    3
    4
    5
    6
    
    <insert id="insert" parameterType="test.model.UserInfo" >
    	<selectKey resultType="java.lang.Integer" keyProperty="id" order="BEFORE">
    	select max(id)+1 from user_info
    	</selectKey>
    	insert user_info <include refid="UserInfoSet" />
    </insert>
    

其中select max(id)+1 from user_info语句是选出表中最大id值+1,并赋予传入的参数userInfo对象,然后再执行insert.
注意到order此时等于before. 应当警惕的是多用户交替插入时,这种方法的安全性,涉及到事务比较麻烦,所以最好还是直接设计数据库时直接设定主键的递增属性。

其他感想:
mybatischoose when好比switch case;
otherwise就是defaultforeach就是传入一个listarraymybatis,然后生成很长的sql
学到这里大概觉得mybatis生成sql,就跟jsp生成html一样,就是输出,就是翻译,编译原理无处不在,重复造轮子无处不在。
暂时不适用trim标签了,太geek了,而且居然只有replace功能,可读性差。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值