数据库的自增主键id(int(11)),查看
auto_increment,记录的是下一次插入时的主键值。(=最大的id+1)1
2
3SHOW TABLE STATUS;
#或:
SELECT AUTO_INCREMENT FROM information_schema.tables WHERE table_name='t_account_0'测试
select @@identity命令:1
2INSERT into user_info (uname, unumber) VALUES ('test','test');
select @@identity as 'curMaxId';注意到第一条命令并没有插入
id值,以便让数据库插入自增id值。
两条命令同一线程中顺序执行后,第二条命令获取到第一条命令插入的id值。
例如,若select @@identity结果为106,则auto_increment此时为107,而INSERT的记录id值为106.第一种mybatis支持自增主键的写法,使用
useGeneratedKeys和keyProperty属性设置: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
6UserInfo userInfo = new UserInfo();
userInfo.setUname("xiaoming");
userInfo.setUnumber("x07");
//a.插入前userinfo的id为null
result = userService.insert(userInfo);
//b.插入后userinfo的id为数据库生成的id值(例如123).第二种使用
selectKey,限定于mysql数据库的写法,其他数据库如oracle为before,各有不同: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的用法:第三种,使用
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. 应当警惕的是多用户交替插入时,这种方法的安全性,涉及到事务比较麻烦,所以最好还是直接设计数据库时直接设定主键的递增属性。
其他感想:
mybatis的choose when好比switch case;otherwise就是default。foreach就是传入一个list或array给mybatis,然后生成很长的sql。
学到这里大概觉得mybatis生成sql,就跟jsp生成html一样,就是输出,就是翻译,编译原理无处不在,重复造轮子无处不在。
暂时不适用trim标签了,太geek了,而且居然只有replace功能,可读性差。