返回主键值的insert操作

应用背景

  • 图示说明

    • 在上述业务背景下,涉及两张数据表的关联操作:用户表 + 用户积分表
    • 传统操作:在对用户表执行完插入语句后,再次查询该用户的uid,将该uid作为外键,最后插入积分数据到用户积分表
      • 数据库操作次数:3次
    • 优化操作:在对用户表执行完插入语句后返回该用户的主键uid,将该uid作为外键,再插入用户积分数据到用户积分表
      • 数据库操作次数:2次
    • 优化操作可以减少对数据库的操作次数

代码实现

UsersMapper.java接口

package com.example.mapper;

import com.example.pojo.User;
import org.apache.ibatis.annotations.Param; import java.util.List; /**
* 数据访问层的接口,定义对数据库完成的CRUD的操作
*/
public interface UsersMapper { //插入用户数据并返回用户id
int insertAndReturnId(User user);
}
  • 接口代码说明

    • 方法中int类型的返回值并不是返回的用户id,而是插入操作执行后受影响的记录条数
    • 至于返回的用户id会通过特殊的sql标签中的属性来指定接收者
    • 后文中的映射文件会具体的介绍该特殊的sql标签及其属性,后文的测试输出结果中,也可以看到该特殊标签及其属性的底层作用原理
    • 在SqlMapConfig.xml核心配置文件中添加日志输出,方便查看sql标签的底层解析
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration PUBLIC "-//mybatis.org//DTD Config 3.0//EN" "http://mybatis.org/dtd/mybatis-3-config.dtd"> <configuration> <!-- 设置日志输出--> <settings>
<setting name="logImpl" value="STDOUT_LOGGING"/>
</settings> </configuration>

UsersMapper.xml映射文件

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <mapper namespace="com.example.mapper.UsersMapper"> <!--
//插入用户数据并返回用户id
int insertAndReturnId(User user);
--> <insert id="insertAndReturnId" parameterType="user">
<selectKey keyProperty="id" resultType="int" order="AFTER">
select
last_insert_id()
</selectKey>
insert into
users(username, birthday, sex, address)
values
(#{userName}, #{birthday}, #{sex}, #{address})
</insert>
</mapper>
  • 映射文件说明

    • 在原先的insert标签中嵌套了selectKey标签(上文提到的特殊标签),关于该子标签的属性说明

      • keyProperty属性:指明子标签的返回结果由入参中的哪个属性来接收
      • resultType属性:子标签的返回结果的类型
      • order属性:子标签相对于外层标签中的sql语句执行的先后顺序
    • last_insert_id():mysql内置的函数,返回插入的最后一条记录的自增主键id值
      • 关于sql标签的执行顺序以及内层标签的返回值到底由谁来接收,可以看测试结果输出的底层内容

测试代码

package com.example.mapper;

import com.example.pojo.User;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.After;
import org.junit.Before;
import org.junit.Test; import java.io.IOException;
import java.io.InputStream;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.List; public class TestUsersMapper { //时间刷
SimpleDateFormat date = new SimpleDateFormat("yyyy-MM-dd"); //SqlSession对象
SqlSession sqlSession; //mybatis动态代理对象
UsersMapper usersMapper; //获取SqlSession
@Before
public void getSqlSession() throws IOException {
//读取核心配置文件
InputStream in = Resources.getResourceAsStream("SqlMapConfig.xml");
//创建SqlSessionFactory对象
SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(in);
//获取SqlSession
sqlSession = factory.openSession();
//获取mybatis动态代理对象
usersMapper = sqlSession.getMapper(UsersMapper.class);
} //归还SqlSession
@After
public void closeSession(){
sqlSession.close();
} @Test
public void testInsertAndReturnId() throws ParseException {
String dataStr = "1996-11-22";
User u = new User("小青", date.parse(dataStr), "女", "沈阳市");
int num = usersMapper.insertAndReturnId(u);
sqlSession.commit();
if(num == 1){
System.out.println("导入数据成功!");
}else{
System.out.println("导入数据失败!");
}
System.out.println("用户id: " + u.getId());
}
}

输出结果

Checking to see if class com.example.mapper.TestUsersMapper matches criteria [is assignable to Object]
Checking to see if class com.example.mapper.UsersMapper matches criteria [is assignable to Object]
Opening JDBC Connection
Created connection 295180183.
Setting autocommit to false on JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@11981797] ==> Preparing: insert into users(username, birthday, sex, address) values (?, ?, ?, ?)
==> Parameters: 小青(String), 1996-11-22 00:00:00.0(Timestamp), 女(String), 沈阳市(String)
<== Updates: 1 ==> Preparing: select last_insert_id()
==> Parameters:
<== Columns: last_insert_id()
<== Row: 30
<== Total: 1 Committing JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@11981797] 导入数据成功!
用户id: 30 Resetting autocommit to true on JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@11981797]
Closing JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@11981797]
Returned connection 295180183 to pool. Process finished with exit code 0

测试结果分析

  • 由sql标签在底层的解析顺序以及解析后的内容可知

    • 在执行映射文件中的标签时,先执行的是selectKey标签外层标签中的sql语句,后执行的是子标签中的sql语句
//先
==> Preparing: insert into users(username, birthday, sex, address) values (?, ?, ?, ?)
==> Parameters: 小青(String), 1996-11-22 00:00:00.0(Timestamp), 女(String), 沈阳市(String)
<== Updates: 1
//后
==> Preparing: select last_insert_id()
==> Parameters:
<== Columns: last_insert_id()
<== Row: 30
<== Total: 1
  • 根据测试代码以及对应的输出结果可知

    • 接口文件中,方法中int类型的返回值并不是返回的用户id,而是插入操作执行后受影响的记录条数

    • keyProperty属性指明了子标签的返回结果(这里是用户id)由入参中的哪个属性来接收

//测试代码

int num = usersMapper.insertAndReturnId(u);
sqlSession.commit();
if(num == 1){
System.out.println("导入数据成功!");
}else{
System.out.println("导入数据失败!");
}
System.out.println("用户id: " + u.getId());
//对应输出结果

导入数据成功!

用户id: 30
  • 数据表中的结果

UUID

  • 由数字字母中划线组成的不重复的36位随机字符串
  • java和mysql对UUID都有支持
package com.example.mapper;

import java.util.UUID;

public class TestUsersMapper {

    @Test
public void testUUID(){
UUID uuid = UUID.randomUUID();
System.out.println(uuid);
}
} /*
输出结果:
c5393056-6711-46c1-8134-80b83d2e9a76 Process finished with exit code 0
*/
select UUID();

--输出结果:13e622a2-1b0f-11ed-b2f9-0242ac110002

mybatis 08: 返回主键值的insert操作 + 利用UUID获取字符串(了解)的更多相关文章

  1. mybatis添加数据时返回主键 insert 返回主键值

    insert 返回主键值 useGeneratedKeys=“true” parameterType=“USer” keyProperty=“id”, <insert id="inse ...

  2. 深入浅出mybatis之返回主键ID

    目录 添加单一记录时返回主键ID 在映射器中配置获取记录主键值 获取新添加记录主键字段值 添加批量记录时返回主键ID 获取主键ID实现原理 添加记录后获取主键ID,这是一个很常见的需求,特别是在一次前 ...

  3. Mybatis批量insert 返回主键值和foreach标签详解

    Mybatis批量insert 返回主键 Mybatis从3.3.1版本开始,支持批量插入后返回主键ID.首先对于支持自增主键的数据库使用useGenerateKeys和keyProperty,对于不 ...

  4. Mybatis+Mysql 返回主键的值

    需求:使用MyBatis往MySQL数据库中插入一条记录后,需要返回该条记录的自增主键值. 方法:在mapper中指定keyProperty属性,示例如下: <insert id="i ...

  5. mybatis+mysql 返回主键

    需求:使用MyBatis往MySQL数据库中插入一条记录后,需要返回该条记录的自增主键值. 方法:在mapper中指定keyProperty属性,示例如下: <insert id="i ...

  6. Mybaits插入记录返回主键值

    某些情况进行insert时不知道主键值(主键为自增),例如系统新增用户时,有用户序号(主键 自增),用户名,密码.插入时只需插入用户名和密码,之后取得mysql自增的序号. 如下为mysql的usr表 ...

  7. mybatis insertUseGeneratedKeys 返回主键为null

    package tk.mybatis.mapper.common.special; import org.apache.ibatis.annotations.InsertProvider; impor ...

  8. MyBatis插入语句返回主键值

    插入语句xml代码: <insert id="insertUser" parameterType="com.spring.mybatis.po.User" ...

  9. 使用mybatis注解@Options实现添加记录时返回主键值

    官网:http://www.mybatis.org/mybatis-3/index.html 在使用mybatis作为ORM框架时,我通常更喜欢使用注解而非xml配置文件的方式.业务场景:添加记录之后 ...

随机推荐

  1. Hadoop入门学习笔记(二)

    Yarn学习 YARN简介 YARN是一个通用资源管理系统和调度平台,可为上层应用提供统一的资源管理和调度 YARN功能说明 资源管理系统:集群的硬件资源,和程序运行相关,比如内存.CPU等. 调度平 ...

  2. 16岁男生信息竞赛成瘾心理出现问题 妈妈:他竟说要AK我

    16岁男生信息竞赛成瘾心理出现问题 -- 妈妈:他竟说要AK我 "我儿子最近快走火入魔了,医生,你救救他吧."40出头的林女士拉着儿子走进江苏省人民医院临床心理科.近几年,信息竞赛 ...

  3. 如何把你的 Android 使用得像 Linux

    前言 最近在学校里上课,老师讲的东西又听不进去,手里只有一个手机和一个平板,之前还可以用 ssh 连接云服务器玩点东西,但是我是用的软件 Juice ssh 并不是很友好,退出到后台一段时间后竟然会自 ...

  4. Halodoc使用 Apache Hudi 构建 Lakehouse的关键经验

    Halodoc 数据工程已经从传统的数据平台 1.0 发展到使用 LakeHouse 架构的现代数据平台 2.0 的改造.在我们之前的博客中,我们提到了我们如何在 Halodoc 实施 Lakehou ...

  5. 使用SSH连接Windows Server 2019 Core

    更新记录 本文迁移自Panda666原博客,原发布时间:2021年7月7日. 一.说明 Windows Server 2019 Core,是纯命令行的Windows Server版本,没有办法使用GU ...

  6. SpringBoot 集成缓存性能之王 Caffeine

    使用缓存的目的就是提高性能,今天码哥带大家实践运用 spring-boot-starter-cache 抽象的缓存组件去集成本地缓存性能之王 Caffeine. 大家需要注意的是:in-memeory ...

  7. SAP APO-需求计划

    需求计划可以对市场中的产品进行预测. 需求计划过程的输出就是需求计划,它考虑了影响需求的所有因素. 需求计划流程定义了需求计划周期中的活动. 由于需求计划过程以循环的形式进行,因此可以重复某些活动. ...

  8. Python自动化办公:批量将文件按分类保存,文件再多,只需一秒钟解决

    序言 (https://jq.qq.com/?_wv=1027&k=GmeRhIX0) 当我们电脑里面的文本或者或者文件夹太多了,有时候想找到自己想要的文件,只能通过去搜索文件名,要是名字忘记 ...

  9. 面试官:Redis 过期删除策略和内存淘汰策略有什么区别?

    作者:小林coding 计算机八股文网站:https://xiaolincoding.com 大家好,我是小林. Redis 的「内存淘汰策略」和「过期删除策略」,很多小伙伴容易混淆,这两个机制虽然都 ...

  10. SpringBoot快速整合通用Mapper

    前言 后端业务开发,每个表都要用到单表的增删改查等通用方法,而配置了通用Mapper可以极大的方便使用Mybatis单表的增删改查操作. 通用mapper配置 1.添加maven: <depen ...