Spring boot整合Mybatis实现级联一对多CRUD操作的完整步骤

前言

在关系型数据库中,随处可见表之间的连接,对级联的表进行增删改查也是程序员必备的基础技能。关于Spring Boot整合Mybatis在之前已经详细写过,不熟悉的可以回顾Spring Boot整合Mybatis并完成CRUD操作,这是本文操作的基础。本文先准备一个测试的数据库,然后使用MyBatis Generator进行部分代码自动生成,再以一个例子来展示稍微高级点的操作:使用Mybatis完成级联一对多的CRUD操作。

数据库准备

数据库用到三张表:user表,role表,user_role表。user表用来存储用户的信息;role表用来存储角色信息;user_role表用来将user和role相关联,存储user和role的映射关系,使得一个用户可以有多个角色,每个角色对应其中的一条记录。

新建user表

CREATE TABLE user(
 id INT PRIMARY KEY AUTO_INCREMENT,
 username VARCHAR(20),
 password VARCHAR(20)
)

新建role表并插入数据

CREATE TABLE role(
 id INT PRIMARY KEY AUTO_INCREMENT,
 rolename VARCHAR(20)
)
INSERT INTO `role`(`rolename`) VALUES ('后台');
INSERT INTO `role`(`rolename`) VALUES ('前端');
INSERT INTO `role`(`rolename`) VALUES ('客户端');
INSERT INTO `role`(`rolename`) VALUES ('AI');
INSERT INTO `role`(`rolename`) VALUES ('大数据');

结果如图:

新建关联表user_role

CREATE TABLE user_role(
 id INT PRIMARY KEY AUTO_INCREMENT,
 userid INT,
 roleid INT
)

自动生成代码

MyBatis Generator 是MyBatis 官方出品的一款代码生成器,为所有版本的MyBatis以及版本2.2.0之后的iBATIS版本生成代码。我们可以使用它自动生成MyBatis的 mapper、dao、entity ,省去最简单的重复代码编写。更多详细情况可以查看官网。

1、pom.xml添加依赖

   <!-- MyBatis Generator插件 -->
   <plugin>
    <groupId>org.mybatis.generator</groupId>
    <artifactId>mybatis-generator-maven-plugin</artifactId>
    <version>1.3.7</version>
    <configuration>
     <verbose>true</verbose>
     <overwrite>true</overwrite>
    </configuration>
    <dependencies>
     <dependency>
      <groupId>mysql</groupId>
      <artifactId>mysql-connector-java</artifactId>
      <version>8.0.16</version>
     </dependency>
    </dependencies>
   </plugin>

2、在项目根目录下面添加自动生成代码的配置文件generatorConfig.xml:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE generatorConfiguration
  PUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN"
  "http://mybatis.org/dtd/mybatis-generator-config_1_0.dtd">
<generatorConfiguration>
 <!--配置文件信息-->
 <properties resource="application.properties"/>

 <!--defaultModelType="flat" 大数据字段,不分表 -->
 <context id="Mysql" targetRuntime="MyBatis3Simple" defaultModelType="flat">
  <property name="autoDelimitKeywords" value="true" />
  <property name="beginningDelimiter" value="`" />
  <property name="endingDelimiter" value="`" />
  <property name="javaFileEncoding" value="utf-8" />
  <plugin type="org.mybatis.generator.plugins.SerializablePlugin" />

  <plugin type="org.mybatis.generator.plugins.ToStringPlugin" />

  <!-- 注释 -->
  <commentGenerator >
   <property name="suppressAllComments" value="true"/><!-- 是否取消注释 -->
   <property name="suppressDate" value="true" /> <!-- 是否生成注释代时间戳-->
  </commentGenerator>

  <!--数据库链接-->
  <jdbcConnection driverClass="${spring.datasource.driverClassName}"
      connectionURL="${spring.datasource.url}"
      userId="${spring.datasource.username}"
      password="${spring.datasource.password}">
  </jdbcConnection>

  <!-- 类型转换 -->
  <javaTypeResolver>
   <!-- 是否使用bigDecimal, false可自动转化以下类型(Long, Integer, Short, etc.) -->
   <property name="forceBigDecimals" value="false"/>
  </javaTypeResolver>

  <!--生成Model类存放位置-->
  <javaModelGenerator targetPackage="com.shangguan.mybatis1.entity" targetProject="src/main/java">
   <property name="enableSubPackages" value="true"/>
   <property name="trimStrings" value="true"/>
  </javaModelGenerator>

  <sqlMapGenerator targetPackage="mapper" targetProject="src/main/resources" >
   <property name="enableSubPackages" value="false" />
  </sqlMapGenerator>

  <javaClientGenerator targetPackage="com.shangguan.mybatis1.dao" targetProject="src/main/java" type="XMLMAPPER" >
   <property name="enableSubPackages" value="false" />
  </javaClientGenerator>

  <!-- 数据库表的信息 -->
  <table tableName="user" enableCountByExample="true" enableUpdateByExample="true" enableDeleteByExample="true" enableSelectByExample="true" selectByExampleQueryId="true">
   <generatedKey column="id" sqlStatement="Mysql" identity="true" />
  </table>

  <table tableName="role" enableCountByExample="true" enableUpdateByExample="true" enableDeleteByExample="true" enableSelectByExample="true" selectByExampleQueryId="true">
   <generatedKey column="id" sqlStatement="Mysql" identity="true" />
  </table>

  <table tableName="user_role" enableCountByExample="true" enableUpdateByExample="true" enableDeleteByExample="true" enableSelectByExample="true" selectByExampleQueryId="true">
   <generatedKey column="id" sqlStatement="Mysql" identity="true" />
  </table>

 </context>
</generatorConfiguration>

3、使用Maven生成代码:

我使用的是Eclipse,工程目录右键Run as --> Maven build,在Goals中输入mybatis-generator:generate命令,点击Run按钮即可自动生成代码。

自动生成代码时遇到的一些坑

报错信息:

Establishing SSL connection without server's identity verification is not recommended.
According to MySQL 5.5.45+, 5.6.26+ and 5.7.6+ requirements SSL
connection must be established by default if explicit option isn't set.
For compliance with existing applications not using
SSL the verifyServerCertificate property is set to 'false'.
You need either to explicitly disable SSL by setting useSSL=false,
or set useSSL=true and provide truststore for server certificate verification.

意思是在说不建议在没有服务器身份验证的情况下建立SSL连接。根据MySQL 5.5.45+、5.6.26+和5.7.6+的要求,如果没有设置显式选项,则默认情况下必须建立SSL连接。您需要通过设置useSSL=false显式禁用SSL,或者设置useSSL=true并为服务器证书验证提供信任存储。

解决方案:在配置文件application.properties中数据库连接后面加上&useSSL=true。

虽然Maven提示“BUILD SUCCESS”,但是仔细看生成的代码和数据库是不匹配的,提示代码被重写,报错[WARNING] Table Configuration user matched more than one table (spring_boot..user,mysql..user,webshop..user,jeece-iyangcong..user)...具体信息如图:

这是因为MySQL 8.0版本驱动将参数nullCatalogMeansCurrent的默认值由true改为了false,在使用MyBatis Generator生成表对应的xml等时会扫描整个服务器里面的全部数据库中的表,而不是扫描对应数据库的表。

解决方案:在配置文件application.properties中数据库连接后面加上&nullCatalogMeansCurrent=true。

如果不出意外的话,将会自动生成3个实体类文件,3个dao层文件,3个mapper.xml。这些代码很长且没有技术含量,在这里就不贴出来的,真有需要可以到文末的GitHub地址去查看。

开始CRUD

接下来需要在Service和ServiceImpl中对dao层进行简单的封装,估计大家都知道该怎么写,在这里也先不贴代码了,详见文末的GitHub地址。

添加用户

添加用户的逻辑是这样的:后台需要分两步处理前端传过来的username,password和roleids。第一步把username和password存入user表;第二步把第一步生成的userid和前端传过来的roleids存入user_role表。这两个操作步骤显然是满足事务的ACID特性的。

Spring 支持编程式事务管理和声明式事务管理两种方式。编程式事务指的是通过编码方式实现事务;声明式事务基于 AOP,将具体业务逻辑与事务处理解耦。声明式事务管理使业务代码逻辑不受污染, 因此在实际使用中声明式事务用的比较多。声明式事务有两种方式,一种是在配置文件(xml)中做相关的事务规则声明,另一种是基于 @Transactional 注解的方式。本文直接使用@Transactional注解实现事务,因为这种方式操作简洁,代码可读性很高。

UserController中增加用户的代码如下:

 @RequestMapping("/addUser")
 @Transactional(rollbackFor={RuntimeException.class, Exception.class})
 public Result saveUser(@RequestParam(value = "username") String username,
   @RequestParam(value = "password") String password,
   @RequestParam(value = "roleids") List<Integer> roleids) {
  User user = new User();
  user.setUsername(username);
  user.setPassword(password);
  userService.addUser(user);
  for(int i=0;i<roleids.size();i++) {
   UserRole userRole = new UserRole();
   userRole.setUserid(user.getId());
   userRole.setRoleid(roleids.get(i));
   userRoleService.addUserRole(userRole);
  }
  return ResultUtils.result(States.errorCode.SUCCESS, "添加成功", null);
 }

使用PostMan测试添加用户:

接口返回添加“添加成功”字样,然后去数据库中查看,user表中多了一条数据,user_role表中也插入了三条数据。显然,这正是需要的结果,这是一个正确的操作。

删除用户

删除用户的逻辑和添加用户的逻辑很相似:第一步根据前端传过来的id删除user表中的记录;第二步userid删除user_role表中的记录;这两个步骤也是满足事务特性,也是使用@Transactional注解来实现。

代码如下:

  @RequestMapping("/deleteUserById")
  @Transactional(rollbackFor={RuntimeException.class, Exception.class})
  public Result deleteUserById(Integer id) {
    userService.deleteUserById(id);
    List<UserRole> list = userRoleService.selectByUserId(id);
    for(int i=0;i<list.size();i++) {
      userRoleService.deleteByPrimaryKey(list.get(i).getId());
    }
    return ResultUtils.result(States.errorCode.SUCCESS, "删除成功", null);
  }

使用PostMan测试删除用户:

接口返回添加“删除成功”字样,然后去数据库中查看,user表中id为1的记录被删除了,user_role表中userid为1的三条记录也都删除了。显然,这正是需要的结果。

修改用户

修改用户逻辑:在user表中修改username和password,同时在user_role表中修改用户和角色的映射记录。修改用户和角色映射记录也就是先按照userid进行删除记录,然后再插入新的映射信息。在这里同样使用@Transactional注解来实现事务。

代码如下:

  @RequestMapping("/updateUser")
  @Transactional(rollbackFor={RuntimeException.class, Exception.class})
  public Result updateUser(@RequestParam(value = "id")Integer id,
      @RequestParam(value = "username") String username,
      @RequestParam(value = "password") String password,
      @RequestParam(value = "roleids") List<Integer> roleids) {
    userService.updateUser(id, username, password);
    //查找user_role然后按照id进行删除
    List<UserRole> list = userRoleService.selectByUserId(id);
    for(int i=0;i<list.size();i++) {
      userRoleService.deleteByPrimaryKey(list.get(i).getId());
    }
    //插入新的roleids
    for(int i=0;i<roleids.size();i++) {
      UserRole record = new UserRole();
      record.setUserid(id);
      record.setRoleid(roleids.get(i));
      userRoleService.addUserRole(record);
    }
    return ResultUtils.result(States.errorCode.SUCCESS, "更新成功", null);
  }

注意:当使用PostMan进行测试的时候,发现报错:org.apache.ibatis.binding.BindingException: Parameter 'username' not found.

Available parameters are [0, 1, 2, param3, param1, param2]。

解决办法:在dao层的Mapper.java代码的参数加上@param注解 。例如:

void updateByPrimaryKey(@Param("id")Integer id,
@Param("username")String username, @Param("password")String
password);

使用PostMan进行测试修改用户:

返回结果没有问题,再去数据库查看,数据库也没有问题,更新操作完成。

查询用户信息

查询用户的信息不仅需要user表中的用户信息,还需要user_role表中的用户角色映射关系,还需要role表的角色信息。这也是需要表之间联合的操作。

本文采用的方法是新建一个AccountDetail类来整合信息。

public class UserDetail {
  private Integer userid;
  private String username;
  private String password;
  private List<Integer> roleids;
  private List<String> rolenames;
  //省略getter和setter
}

这是整合信息的关键代码:

public List<UserDetail> selectAll(){
  List<User> userList = new ArrayList<>();
  List<UserDetail> details = new ArrayList<>();

  try{
    userList = userMapper.selectAll();
    details = getUserDetails(userList);
  }catch(Exception e){
    e.printStackTrace();
    return details;
  }
  return details;
}

public List<UserDetail> getUserDetails(List<User> lists){
  List<UserDetail> details = new ArrayList<>();
  if(lists == null || lists.size() < 1){
    return details;
  }
  Map<Integer, String> nameMap = roleService.getNameMap();
  for(int i=0; i< lists.size();i++){
    User user = lists.get(i);
    UserDetail detail = new UserDetail();

    detail.setUserid(user.getId());
    detail.setUsername(user.getUsername());
    detail.setPassword(user.getPassword());
    List<Integer> roleids = new ArrayList<>();
    List<String> rolenames = new ArrayList<>();
    List<UserRole> userroles = userRoleMapper.selectByUserId(user.getId());
    for(int j=0;j<userroles.size();j++) {
      roleids.add(userroles.get(j).getRoleid());
      rolenames.add(nameMap.get(userroles.get(j).getRoleid()));
    }
    detail.setRoleids(roleids);
    detail.setRolenames(rolenames);
    details.add(detail);
  }
  return details;
}

这是封装的接口:

@RequestMapping("/getAllUser")
public Result getAllUser() {
  List<UserDetail> list = userService.selectAll();
  return ResultUtils.result(States.errorCode.SUCCESS, "查询成功", list);
}

使用PostMan进行测试查询用户信息:

可以看到这个接口返回了所有的用户信息,包括用户的基本信息和角色信息,准确无误。

总结

Spring和MyBatis实现一对多关联的增删改查也有多种方式:可以使用MyBatis来自定义SQL语句来实现;也可以使用Spring的注解结合MyBatis自动生成的代码来实现。我更喜欢后者,因为通过Mybatis Generator自动生成代码以后,这些代码就不需要再修改了,可以直接封装service和controller。希望本文对大家有用。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对我们的支持。

(0)

相关推荐

  • springboot+springmvc+mybatis项目整合

    介绍: 上篇给大家介绍了ssm多模块项目的搭建,在搭建过程中spring整合springmvc和mybatis时会有很多的东西需要我们进行配置,这样不仅浪费了时间,也比较容易出错,由于这样问题的产生,Pivotal团队提供了一款全新的框架,该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置.通过这种方式,Spring Boot致力于在蓬勃发展的快速应用开发领域(rapid application development)成为领导者. 特点: 1. 创建独立的Spring应用

  • springboot与mybatis整合实例详解(完美融合)

    简介 从 Spring Boot 项目名称中的 Boot 可以看出来,Spring Boot 的作用在于创建和启动新的基于 Spring 框架的项目.它的目的是帮助开发人员很容易的创建出独立运行和产品级别的基于 Spring 框架的应用.Spring Boot 会选择最适合的 Spring 子项目和第三方开源库进行整合.大部分 Spring Boot 应用只需要非常少的配置就可以快速运行起来. Spring Boot 包含的特性如下: 创建可以独立运行的 Spring 应用. 直接嵌入 Tomc

  • Mybatis中的高级映射一对一、一对多、多对多

    学习hibernate的时候,小编已经接触多各种映射,mybatis中映射有到底是如何运转的,今天这篇博文,小编主要来简单的介绍一下mybatis中的高级映射,包括一对一.一对多.多对多,希望多有需要的小伙伴有帮助,小编主要从四个方面进行介绍,订单商品数据模型.一对一查询.一对多查询.多对多查询. 一.订单商品数据模型 1.数据库执行脚本,如下所示: <span style="font-family:Comic Sans MS;font-size:18px;">CREATE

  • Mybatis 中的一对一,一对多,多对多的配置原则示例代码

    什么是 MyBatis ? MyBatis 是支持定制化 SQL.存储过程以及高级映射的优秀的持久层框架.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis 可以对配置和原生Map使用简单的 XML 或注解,将接口和 Java 的 POJOs(Plain Old Java Objects,普通的 Java对象)映射成数据库中的记录. 表:market_plan(营销计划(关联了用户)) market_plan_product(产品关联营销计划) mark

  • mybatis一对多查询功能

    首先,我们还是先给出一个需求:根据订单id查询订单明细--我们知道,一个订单里面可以有多个订单的明细(需求不明确的同学,请留言或者去淘宝网上的订单处点一下就知道了).这个时候,一个订单,对应多个订单的id.这种需求出现的时候,我们应该如何查询呢? 此时我们的数据模型如下图(左)由于查询用户也是我们的需求,所以就在原有的基础上进行扩展,数据模型如下(右): 很显然,如果用resultType的方式去实现的话,是不合理的了.因为我们需要创建一个既有订单又有订单明细的pojo然后呢,我们的mybati

  • MyBatis之自查询使用递归实现 N级联动效果(两种实现方式)

    A:首先先看下一个简单的面试题 斐波那契数列 计算数组{1,1,2,3,5,8.......} 第30位值 规律:1 1 从第三项开始,每一项都是前两项之和 有两种实现方式  第一种方式: public class TestOne { public int TestSelf(int n){ if(n<0){ throw new IllegalArgumentException("n不能为负数"); }else if(n<=2){ return 1; }else{ retur

  • Mybatis 一对多和多对一关联查询问题

    首先  数据库量表之间字段关系(没有主外键) studentmajor表的id字段对应student表里major字段 两个实体类 package com.model; import java.util.Date; public class Student { private Integer sno; private String sname; private String ssex; private Integer sclass; private StudentMajor studentmaj

  • MyBatis存储过程、MyBatis分页、MyBatis一对多增删改查操作

    一.用到的实体类如下: Student.java package com.company.entity; import java.io.Serializable; import java.util.Date; public class Student implements Serializable{ private static final long serialVersionUID = 1L; private int id; private String name; private Date

  • Spring Boot整合MyBatis操作过程

    1.加入mybatis-spring-boot-stater的Maven依赖 <dependency> <groupId>org.mybatis.spring.boot</groupId> <artifactId>mybatis-spring-boot-starter</artifactId> <version>1.1.1</version> </dependency> 2.配置数据源 在src/main/re

  • Spring boot Mybatis 整合(完整版)

    本项目使用的环境: 开发工具: Intellij IDEA 2017.1.3 springboot: 1.5.6 jdk:1.8.0_161 maven:3.3.9 额外功能 PageHelper 分页插件 mybatis generator 自动生成代码插件 步骤: 1.创建一个springboot项目: 2.创建项目的文件结构以及jdk的版本 3.选择项目所需要的依赖 然后点击finish 5.看一下文件的结构: 6.查看一下pom.xml: <?xml version="1.0&qu

随机推荐