在Mybatis中,执行insert操作时,如果我们希望返回数据库生成的自增主键值,那么就需要使用到KeyGenerator对象。
需要注意的是,KeyGenerator的作用,是返回数据库生成的自增主键值,而不是生成数据库的自增主键值。返回的主键值放到哪儿呢?放到parameter object的主键属性上。
下面看看其接口定义。
data:image/s3,"s3://crabby-images/aa977/aa97755b3530a5e3143444b8eed7647e81442fa3" alt=""
接口定义还是比较简单的,就是在insert前、insert后,策略处理主键值。
Jdbc3KeyGenerator:用于处理数据库支持自增主键的情况,如MySQL的auto_increment。
NoKeyGenerator:空实现,不需要处理主键。
SelectKeyGenerator:用于处理数据库不支持自增主键的情况,比如Oracle的sequence序列。
上面都比较泛泛而谈,我们来点实际的,看看它们都是如何工作的。
1. JDBC实现insert后,返回自增主键值的原理
data:image/s3,"s3://crabby-images/36534/365344ef8fa66d7751e8703ba0d4c203604791da" alt=""
以上代码,仅作为演示使用。Mybatis是对JDBC的封装,其Jdbc3KeyGenerator类,就是使用上面的原理,来返回数据库生成的主键值的。
2. Jdbc3KeyGenerator源码解读
data:image/s3,"s3://crabby-images/35065/35065bff16c7e8ebd087b2e2ccefeeab278ebb9b" alt=""
Mapper.Xml配置方式。
data:image/s3,"s3://crabby-images/20f77/20f77543bf47060bae7c54936f4a602f40b37cca" alt=""
3. NoKeyGenerator源码解读
完全是空实现,没啥可说的。
4. SelectKeyGenerator的原理
data:image/s3,"s3://crabby-images/20027/20027818b4fc572a6216cac5443ec61219216fc8" alt=""
在执行insert之前,先发起一个sql查询,将返回的序列值赋值给Student的stuId属性,然后再执行insert操作,这样表中的stud_id字段就有值了。order="BEFORE"表示insert前执行,比如取sequence序列值;order="AFTER"表示insert之后执行,比如使用触发器给主键stud_id赋值。比较简单,我就不再贴源码了。
注意:由于selectKey本身返回单个序列主键值,也就无法支持批量insert操作并返回主键id列表了。如果要执行批量insert,请选择使用for循环执行多次插入操作。
5. KeyGenerator的创建过程
每一个MappedStatement,都有一个非空的KeyGenerator引用。
org.apache.ibatis.mapping.MappedStatement.Builder.Builder()构造方法赋初始值源码。
data:image/s3,"s3://crabby-images/245fc/245fc7584a7f9412456931275478edf4715866ad" alt=""
org.apache.ibatis.builder.xml.XMLStatementBuilder.parseStatementNode()覆盖KeyGenerator初始值的源码。
data:image/s3,"s3://crabby-images/d3fa9/d3fa91f8e8e2084d5fc72c9eb0fea746b31bf899" alt=""
org.apache.ibatis.builder.xml.XMLStatementBuilder.parseSelectKeyNode()解析元素,构建SelectKeyGenerator的源码。
data:image/s3,"s3://crabby-images/c655f/c655fdf7722991df5323b65d5c525914e253127c" alt=""
因此,只有SelectKeyGenerator会保存至Configuration对象的Map<String, KeyGenerator> keyGenerators属性当中。元素,会被Mybatis解析为一个MappedStatement对象,并作为构造参数传递至SelectKeyGenerator内保存起来。
data:image/s3,"s3://crabby-images/8e325/8e32577b8afb958ea11f8de1515879e6bb60b703" alt=""
Map<String, KeyGenerator> keyGenerators的存储结构如下。
data:image/s3,"s3://crabby-images/1976f/1976fa57019d270d12e99c871a2f6b58a66d8c77" alt=""
至此,每一个MappedStatement对象,都恰当的绑定了一个KeyGenerator对象,就可以开始工作了。
6. KeyGenerator的使用过程
keyGenerator.processBefore()方法调用时机。
org.apache.ibatis.executor.statement.BaseStatementHandler.BaseStatementHandler()构造方法源码。
data:image/s3,"s3://crabby-images/8404c/8404c1b2a7042aa6972be24349327fc6d1518a0a" alt=""
即,创建StatementHandler对象时,就会执行keyGenerator.processBefore()方法。keyGenerator.processAfter()方法,自然就是Statement执行后执行了。
org.apache.ibatis.executor.statement.SimpleStatementHandler.update(Statement)方法源码。其他的StatementHandler都是类似的。
data:image/s3,"s3://crabby-images/0fe33/0fe3397860cecbc4e0f0d6a7edaaacd3c3b9ac91" alt=""
7. 批量插入,返回主键id列表
data:image/s3,"s3://crabby-images/abe72/abe7219861df0ab4c427005f56fdf3d775f5bb0d" alt=""
对的,你没看错,就是像上面这样for循环逐一insert操作的,此时,如果你考虑性能的话,可以使用BatchExecutor来完成,当然了,其他的Executor也是可以的。
如果文章就像上面这样写,那么就完全失去了写文章的价值,上面的for循环,谁都懂这么操作可以实现,但是,很多人想要的并不是这个例子,而是另外一种批量插入操作,返回主键id列表。那么,看第8条。
8. Mybatis批量插入,返回主键id列表为null
data:image/s3,"s3://crabby-images/5d48b/5d48b2bfbd12da1096de71c58519a74b46e270b9" alt=""
很多同学,包括开源中国社区,都遇到使用上面的批量insert操作,返回的主键id列表是null的问题,很多人得出结论:Mybatis不支持这种形式的批量插入并返回主键id列表。真是这样吗?
我必须明确的跟大家说,Mybatis是支持上述形式的批量插入,且可以正确返回主键id列表的。之所以返回null值,是Mybatis框架的一个bug,下一篇将具体讲述产生这个bug的原因,以及如何修复它。
网友评论