(六) MyBatis从入门到入土——主键的获取

这是mybatis系列第六篇。没看前文的建议先去【Java冢狐】公众号中查看前文,方便理解和掌握。

前面五篇把MyBatis的基本用法基本介绍完毕了,强烈推荐大家先把前面五篇看完实际操作完再来看接下来的内容。

增删改返回值说明

mybatis中对db执行增删改操作,不管是新增、删除、还是修改,其本质都是使用jdbc中对应的方法,即java.sql.Statement的executeUpdate的方法,或者java.sql.PreparedStatement的executeUpdate方法。

所以问题的关键在于掌握这两个方法的内容,而这两个方法的参数不一致但是返回值都是int,用来表示影响的行数。

而Mybatis的返回值再支持int的基础上,还支持Integer、long、Long、boolean、Boolean、void多种类型,可谓是大大加强了编程的灵活性和多样性。

但是聪明的小伙伴肯定知道,MyBatis能支持这么多种的类型,其实就是mybatis在内部将int类型转换成其他类型而已。没有什么高深的东西、

了解了增删改查的返回值后,下一步我们再去看一看jdbc和mybatis是如何获取主键的。

jdbc获取主键的几种方式

在聊jdbc获取主键的方式之前,我们先来回想一下数据库的主键是怎么样生成的。

那mysql举例:在mysql中当我们向表中插入数据的时候,如果不指定主键(id),那么mysql会自动生成id,但是在一些业务中我们后续会需要这个id,此时我们就要想办法去获取这个id,尤其是这种自动生成的自增id。

要想要清楚的了解,就需要先看看jdbc是如何实现的

方式1:jdbc内置的方式

用法

获取主键的需求,jdbc的开发人员就已经想到了,并且也开发了相应的api来帮助我们编程,具体看 java.sql.Statement类中的getGeneratedKeys方法

从这个方法中我们得知这个方法会返回一个结果集,从这个结果集中可以获取自增主键的值。但是在使用这个方法有个前提,那就是执行sql的时候需要做一个设置。

如果是通过java.sql.Statement执行sql,需要调用下面这个方法:

int executeUpdate(String sql, int autoGeneratedKeys) throws SQLException

注意上面这个方法的第二个参数需要设置为java.sql.Statement.RETURN_GENERATED_KEYS,表示需要返回自增列的值。

不过多数情况下,我们会使用java.sql.PreparedStatement对象来执行sql,如果想获取自增值,创建这个对象需要设置第2个参数的值,如下:

PreparedStatement preparedStatement = connection.prepareStatement(sql, Statement.RETURN_GENERATED_KEYS);

然后我们就可以通过getGeneratedKeys返回的ResultSet对象获取自动增长的值了,如下:

ResultSet generatedKeys = preparedStatement.getGeneratedKeys();
if (generatedKeys!=null && generatedKeys.next()) {
    log.info("自增值为:{}", generatedKeys.getInt(1));
}

方式2:插入之后查询获取

用法

当然除了jdbc想到了我们这个需求,mysql数据库也想到了这个需求也有对应的语句来帮助我们在插入数据后获取最新插入记录的id值,如下所示

SELECT LAST_INSERT_ID()

所以我们可以在插入之后,立即使用当前连接发送上面这条sql去获取自增列的值就可以。这种方法相对来说十分的简单和方便,但是也存在着一些问题,就是在高并发多线程的情况 ,可能会出现严重的问题,所以一般来说不建议使用。

方式3:插入之前获取

最后一种方法就是在插入之前获取。

oracle不知道大家有没有玩过,oracle中没有mysql中自动增长列,但是oracle有个功能可以实现自动增长,这个功能就是序列,序列就相当于一个自增器一样,有个初始值,每次递增的步长,当然这个序列提供了一些功能给我们使用,可以获取序列的当前值、下一个值,使用方式如下:

  • 先定义一个序列
  • 获取下一个值:SELECT 序列名.NEXTVAL FROM dual;

mybatis获取主键的3种方式

既然jdbc有三种方法,那么mybatis也有相应的几种方法。

方式1:内部使用jdbc内置的方式

用法

mybatis这个方式内部采用的是上面说的jdbc内置的方式。

我们需要在Mapper xml中进行配置,如:

<insert id="insertUser1" parameterType="zhonghu.mybatis.chat01.UserModel" useGeneratedKeys="true" keyProperty="id">
    <![CDATA[
    INSERT INTO user (name,age,salary) VALUES (#{name},#{age},#{salary})
     ]]>
</insert>

有2个关键参数必须要设置:

  • useGeneratedKeys:设置为true
  • keyProperty:参数对象中的属性名称,最后插入成功之后,mybatis会通过反射将自增值设置给keyProperty指定的这个属性

方式2:插入后查询获取主键

用法

这个方式和上面介绍的jdbc的第二种方式一样,插入之后通过查询获取主键的值然后填充给指定的属性,mapper xml配置如下:

<insert id="insertUser2" parameterType="zhonghu.mybatis.chat01.UserModel">
    <selectKey keyProperty="id" order="AFTER" resultType="long">
    <![CDATA[
    SELECT LAST_INSERT_ID()
     ]]>
    </selectKey>
    <![CDATA[
    INSERT INTO user (name,age,salary) VALUES (#{name},#{age},#{salary})
     ]]>
</insert>

关键代码是selectKey元素包含的部分,这个元素内部可以包含一个sql,这个sql可以在插入之前或者插入之后运行(之前还是之后通过order属性配置),然后会将sql运行的结果设置给keyProperty指定的属性,selectKey元素有3个属性需要指定:

  • keyProperty:参数对象中的属性名称,最后插入成功之后,mybatis会通过反射将自增值设置给keyProperty指定的这个属性
  • order:指定selectKey元素中的sql是在插入之前运行还是插入之后运行,可选值(BEFORE|AFTER),这种方式中我们选择AFTER
  • resultType:keyProperty指定的属性对应的类型,如上面的id对应的类型是java.lang.Long,我们直接写的是别名long

方式3:插入前查询获取主键

用法

这个方式和上面介绍的jdbc的第3种方式一样,会在插入之前先通过一个查询获取主键的值然后填充给指定的属性,然后在执行插入,mapper xml配置如下:

<insert id="insertUser3" parameterType="zhonghu.mybatis.chat01.UserModel">
    <selectKey keyProperty="id" order="BEFORE" resultType="long">
    <![CDATA[ 获取主键的select语句 ]]>
    </selectKey>
    <![CDATA[
    INSERT INTO user (name,age,salary) VALUES (#{name},#{age},#{salary})
     ]]>
</insert>

关键代码是selectKey元素包含的部分,这个元素内部可以包含一个sql,这个sql可以在插入之前或者插入之后运行(之前还是之后通过order属性配置),然后会将sql运行的结果设置给keyProperty指定的属性,selectKey元素有3个属性需要指定:

  • keyProperty:参数对象中的属性名称,最后插入成功之后,mybatis会通过反射将自增值设置给keyProperty指定的这个属性
  • order:指定selectKey元素中的sql是在插入之前运行还是插入之后运行,可选值(BEFORE|AFTER),这种方式中我们选择BEFORE
  • resultType:keyProperty指定的属性对应的类型,如上面的id对应的类型是java.lang.Long,我们直接写的是别名long

总结

本篇文章着重介绍了mybatis中一些增删改查的高级使用方法,尤其是获取主键的方法,在此进行了jdbc和mybatis对比介绍了三种方法。

最后

  • 如果觉得看完有收获,希望能关注一下,顺便给我点个赞,这将会是我更新的最大动力,感谢各位的支持
  • 欢迎各位关注我的公众号【java冢狐】,专注于java和计算机基础知识,保证让你看完有所收获,不信你打我
  • 求一键三连:点赞、转发、在看。
  • 如果看完有不同的意见或者建议,欢迎多多评论一起交流。感谢各位的支持以及厚爱。

——我是冢狐,和你一样热爱编程。

欢迎关注公众号“Java冢狐”获取最新消息

(0)

相关推荐

  • 小师妹问我:Mybatis常见注解有哪些?

    回复"面试"获取全套面试资料 当下,注解非常流行,以前很长篇的代码,现在基本上一个注解就能搞定. 那,在Mybatis中又有哪些注解呢? Mybatis中的注解基本上都在org.a ...

  • (二)MyBatis从入门到入土——开发一个Mybatis项目

    这是mybatis系列第2篇.没看前文的建议先去[Java冢狐]公众号中查看前文,方便理解和掌握这篇文章主要接着前文介绍了如何创建并使用Mybatis. 实战演练 上一篇文章中我们大体介绍了MyBat ...

  • (三)MyBatis从入门到入土——使用详解

    MyBatis使用详解 上篇我们手动开发了一个MyBatis项目,但是我们仅仅是编写了代码,对于整个项目是如何运行以及每个代码的意义都没有仔细的分析和说明,那么接下来我们就开始分析每个代码的意义以及如 ...

  • (四)Mybatis从入门到入土——别名、配置文件以及引入mapper

    这是mybatis系列第4篇.没看前文的建议先去[Java冢狐]公众号中查看前文,方便理解和掌握 别名 为什么需要使用别名? 在xml文件中有很多需要类完整的类名的地方,十分的冗长,为了减轻我们的工作 ...

  • (五)Mybatis从入门到入土——Mapper接口传参多种方式解析

    这是mybatis系列第5篇. 说到底Mybatis常见的传参形式无非是传递一个参数.Map.Java对象,亦或是多个参数.下面就分别对这些进行讲解和说明. 传递一个参数 传递一个参数相对来说较为简单 ...

  • (八)MyBatis从入门到入土——自动映射的使用

    今天我们介绍一下我们一直在使用的但是没有来得及做过多介绍的自动映射. 什么是自动映射? 介绍自动映射之前先看一下手动映射,如下: <resultMap id="orderModelMa ...

  • (六)Spring从入门到入土——Bean的装配机制

    Spring Bean的装配机制 Spring中bean有三种装配机制,分别是: 在xml中显式配置: 隐式的bean发现机制和自动装配. 在java中显式配置:(java Config) Sprin ...

  • 669期B || 妥妥读《三国》(六十八)常欺主孙綝作死 强反击曹髦被杀/轩诚播读

    美丽诗文 | 精品连播 | 美丽杂谈 | 艺术空间 | 经典时刻 上期结尾: 全纪的母亲赶紧向自己的弟弟孙綝通报了情况下.孙綝知道后,就把孙亮这个皇帝给废掉了.另立了一个叫做孙休的人当了吴国皇帝.孙休 ...

  • 围棋定式:入门到入土(85)

    我再不更新,大家可能就已经忘了我并不喜欢打中单了.我只是个臭酱油,给AI包鸡包眼包大药那种. 有时候,你永远不知道自己坑会踩在何处.这个系列,想介绍一些从未体验过的定式大坑.本期开始将陆续介绍小目守角 ...

  • 围棋定式:入门到入土(86)

    我真不是忘了这个系列. 我就是懒. 有时候,你永远不知道自己坑会踩在何处.这个系列,想介绍一些从未体验过的定式大坑.本期开始将陆续介绍小目守角的各种变化,这应该是实战中非常实用的系列. 图1 对于黑1 ...