MybatisPlus基本使用、多表查询和分页查询

MybatisPlus基本使用、多表查询和分页查询-ALOOGY WORLD (aloogychendl23.asia)

一、MyBatis的基本使用

数据持久化概述

1. 什么是持久化?
  • 持久化(Persistence),即把数据(如内存中的对象)保存到可永久保存的存储设备中(如磁盘)。

  • 持久化的主要应用是将内存中的数据存储在关系型的数据库中,当然也可以存储在磁盘文件中、XML数据文件中等。

2. ORM原理
  • 即Object-Relationl Mapping,它的作用是在关系型数据库和对象之间作一个映射,这样在具体的操作数据库的时候,就不需要再去和复杂的SQL语句打交道,只要像平时操作对象一样操作它就可以了 。

3. 常用的ORM框架
  • Hibernate:全自动需要些hql语句

  • iBATIS:半自动自己写sql语句,可操作性强,小巧

  • MyBatis:是一个半自动的ORM持久层框架,内部封装了JDBC。

  • MyBatisPlus:是一个Mybatis的增强工具,它在Mybatis的基础上做了增强。

4. Mybatis简介
4.1原始jdbc操作(查询数据)

image-20240516164418988

4.2原始jdbc操作(插入数据)

image-20240516160306475

4.3 原始jdbc操作的分析

原始jdbc开发存在的问题如下:

①数据库连接创建、释放频繁造成系统资源浪费从而影响系统性能

②sql 语句在代码中硬编码,造成代码不易维护,实际应用 sql 变化的可能较大,sql 变动需要改变java代码。

③查询操作时,需要手动将结果集中的数据手动封装到实体中。插入操作时,需要手动将实体的数据设置到sql语句的占位符位置

应对上述问题给出的解决方案:

①使用数据库连接池初始化连接资源

②将sql语句抽取到xml配置文件中

③使用反射、内省等底层技术,自动将实体与表进行属性与字段的自动映射

4.4 什么是Mybatis

mybatis 是一个优秀的基于java的持久层框架,它内部封装了jdbc,使开发者只需要关注sql语句本身,而不需要花费精力去处理加载驱动、创建连接、创建statement等繁杂的过程。

mybatis通过xml或注解的方式将要执行的各种 statement配置起来,并通过java对象和statement中sql的动态参数进行映射生成最终执行的sql语句。

最后mybatis框架执行sql并将结果映射为java对象并返回。采用ORM思想解决了实体和数据库映射的问题,对jdbc 进行了封装,屏蔽了jdbc api 底层访问细节,使我们不用与jdbc api 打交道,就可以完成对数据库的持久化操作。

环境准备

1. 数据库准备

构建数据表,用户表和订单表的关系为,一个用户有多个订单,一个订单只从属于一个用户

 CREATE TABLE `user` (
   `id` int(11) NOT NULL AUTO_INCREMENT,
   `username` varchar(50) DEFAULT NULL,
   `password` varchar(50) DEFAULT NULL,
   `birthday` varchar(50) DEFAULT NULL,
   PRIMARY KEY (`id`)
 ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
 -- 数据----------------------------
 INSERT INTO `user` VALUES ('1', 'lucy', '123', '2018-12-12');
 INSERT INTO `user` VALUES ('2', 'haohao', '123', '2019-12-12');
 DROP TABLE IF EXISTS `orders`;
 CREATE TABLE `orders` (
   `id` int(11) NOT NULL AUTO_INCREMENT,
   `ordertime` varchar(255) DEFAULT NULL,
   `total` double DEFAULT NULL,
   `uid` int(11) DEFAULT NULL,
   PRIMARY KEY (`id`)
 ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
 -- 数据-----------------------------
 INSERT INTO `orders` VALUES ('1', '2018-12-12', '3000', '1');
 INSERT INTO `orders` VALUES ('2', '2018-12-12', '4000', '1');
 INSERT INTO `orders` VALUES ('3', '2018-12-12', '5000', '2');
2. 添加依赖
 <!--  MyBatisPlus依赖  -->
 <dependency>
     <groupId>com.baomidou</groupId>
     <artifactId>mybatis-plus-boot-starter</artifactId>
     <version>3.4.2</version>
 </dependency>
 <!-- mysql驱动依赖  -->
 <dependency>
      <groupId>mysql</groupId>
      <artifactId>mysql-connector-java</artifactId>
      <version>5.1.47</version>
 </dependency>
 <!--    数据连接池 druid    -->
 <dependency>
      <groupId>com.alibaba</groupId>
      <artifactId>druid-spring-boot-starter</artifactId>
      <version>1.1.20</version>
 </dependency>
3. 添加全局配置
 spring.datasource.type=com.alibaba.druid.pool.DruidDataSource
 spring.datasource.driver-class-name=com.mysql.jdbc.Driver
 spring.datasource.url=jdbc:mysql://localhost:3306/mydb?useSSL=false
 spring.datasource.username=root
 spring.datasource.password=root
 mybatis-plus.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl
 ​
4. 添加@MapperScan 注解
  • 在 Spring Boot 启动类中添加 @MapperScan 注解,扫描 Mapper 文件夹

 @SpringBootApplication
 @MapperScan("com.xx.mapper")
 public class MybatisplusDemoApplication {
     public static void main(String[] args) {
         SpringApplication.run(MybatisplusDemoApplication.class, args);
     }
 }

Mybatis的注解开发

1. MyBatis的常用注解

我们先围绕一些基本的CRUD来学习,再学习复杂映射多表操作, MyBatis用于数据表CRUD操作常见注解如下:

注解功能
@Insert实现插入
@Update实现更新
@Delete实现删除
@Select实现查询
@Result实现结果集封装
@Results可以与@Result 一起使用,封装多个结果集
@One实现一对一结果集封装
@Many实现一对多结果集封装
2. Mapper实现

接下来完成简单的user表的增删改查的操作

编写实体类User.java

 public class User {   
     private int id;
     private String username;
     private String password;
     private String birthday;
 }

编写UserMapper接口,加入以下增删改查方法

 @Mapper
 public interface UserMapper {
     @Insert("insert into user values(#{id},#{username},#{password},#{birthday})")
     int add(User user);
 ​
     @Update("update user set username=#{username},password=#{password},birthday=#{birthday} where id=#{id}")
     int update(User user);
 ​
     @Delete("delete from user where id=#{id}")
     int delete(int id);
 ​
     @Select("select * from user where id=#{id}")
     User findById(int id);
 ​
     @Select("select * from user")
     List<User> getAll();
 }
3. 控制器实现

创建UserController控制器,在控制器中注入UserMapper实例,注意!一定要添加第3节中的@MapperScan注解,否则无法注入mapper实例。

 public class UserController {
 ​
     @Autowired  //自动注入,Spring会自动实例化UserMapper,并注入此属性中(依赖注入)
     private UserMapper userMapper;
     
 }

添加控制器方法,测试增删改查操作。

 //  添加用户:localhost:8080/add?id=100&username=zhangsan&password=123&birthday=2020-11-12
     @GetMapping("add")
     public String add(User user){
         int i = userMapper.add(user);
         System.out.println(user);
         if(i > 0){
             return "添加成功";
         }else{
             return "添加失败";
         }
     }
 //  更新用户:localhost:8080/update?id=100&username=lisi&password=123&birthday=2020-11-12
     @GetMapping("update")
     public  String update(User user){
         int i = userMapper.update(user);
         if(i > 0){
             return "更新成功";
         }else{
             return "更新失败";
         }
     }
 ​
 //  删除用户:localhost:8080/del?id=100
     @GetMapping("del")
     public String del(int id){
         int i = userMapper.delete(id);
         if(i > 0){
             return "删除成功";
         }else{
             return "删除失败";
         }
     }
 ​
 //  查询单个用户:localhost:8080/find?id=1
     @GetMapping("find")
     public User findById(int id){
         User user = userMapper.findById(id);
         return user;
     }
 //  查询所有用户:localhost:8080/findAll
     @GetMapping("findAll")
     public List findAll(){
         return userMapper.getAll();
     }

MyBatis的注解实现复杂映射开发

实现复杂关系映射,可以使用@Results注解,@Result注解,@One注解,@Many注解组合完成复杂关系的配置

注解说明
@Results代替<resultMap>标签,该注解中可以加入单个或多个@Result注解
@Result代替<id>标签和<Result>标签,@Result中可以使用以下属性: - column:数据表的字段名称 - property: 类中对应的属性名 - one: 与@One注解配合,进行一对一的映射 - many: 与@Many注解配合,进行一对多的映射
@One代替<assocation>标签,用于指定查询中返回的单一对象 通过select属性指定用于多表查询的方法 使用格式:@Result(column="",property="",one=@One(select=""))
@Many代替<collection>标签,用于指定查询中返回的集合对象 使用格式:@Result(column="",property="",many=@Many(select=""))
1. 一对一查询模型

用户表和订单表的关系为,一个用户有多个订单,一个订单只从属于一个用户

一对一查询的需求:查询一个订单,与此同时查询出该订单所属的用户

对应的sql语句:

 select * from orders;
 select * from user where id=查询出订单的uid;

创建Order和User实体

 public class Order {
     private int id;
     private Date ordertime;
     private double total;
     //代表当前订单从属于哪一个客户
     private User user;
 }
 public class User {   
     private int id;
     private String username;
     private String password;
     private String birthday;
 }

使用注解配置OrderMapper

 @Mapper
 public interface OrderMapper {
     @Select("select * from orders")
     @Results({
             @Result(column = "id",property = "id"),
             @Result(column = "ordertime",property = "ordertime"),
             @Result(column = "total",property = "total"),
             @Result(column = "uid",property = "user",javaType = User.class,
                 one = @One(select = "com.xx.mapper.UserMapper.findById")
             ),
     })
     List<Order> findAll();
 }

UserMapper

 public interface UserMapper {
     @Select("select * from user where id=#{id}")
     User findById(int id);
 }

创建OrderController测试

 @RestController
 public class OrderController {
 ​
     @Autowired
     private OrderMapper orderMapper;
 ​
     @GetMapping("order")
     public List<Order> find(){
 ​
         return orderMapper.findAll();
     }
 }
2. 一对多查询模型

用户表和订单表的关系为,一个用户有多个订单,一个订单只从属于一个用户

一对多查询的需求:查询一个用户,与此同时查询出该用户具有的订单

对应的sql语句:

 select * from user;
 select * from orders where uid=查询出用户的id;

修改User实体

 public class Order {
     private int id;
     private String ordertime;
     private double total;
     //代表当前订单从属于哪一个客户
     private User user;
 }
 public class User {   
     private int id;
     private String username;
     private String password;
     private String birthday;
     //代表当前用户具备哪些订单
     private List<Order> orders;
 }

使用注解配置Mapper

 public interface UserMapper {
     @Select("select * from user")
     @Results({
             @Result(column = "id",property = "id"),
             @Result(column = "username",property = "username"),
             @Result(column = "password",property = "password"),
             @Result(column = "id",property = "orders",javaType = List.class,
                 many=@Many(select = "com.xx.mapper.OrderMapper.findByUid")
             )
     })
     List<User> findAllUserAndOrder();
 }
 ​
 public interface OrderMapper {
     @Select("select * from orders where uid=#{uid}")
     List<Order> findByUid(int uid);
 }

在UserController中测试

 //  查询所有用户:localhost:8080/findAll
 @GetMapping("findAll")
 public List findAll(){
     return userMapper.findAllUserAndOrder();
 }

二、多表查询(关联查询)

实现复杂关系映射,可以使用@Results注解,@Result注解,@One注解, @Many注解(MybatisPlus)组合完成复杂关系的配置。

注解说明
@Results代替<resultMap>标签,该注解中可以加入单个或多个@Result注解
@Result代替<id><Result>标签,@Result中可以使用以下属性: -column:数据表的字段名称 -property:类中对应的属性名 -one:与@One注解配合,进行一对一的映射 -many:与@Many注解配合,进行一对多的映射
@One代替<assocation>标签,用于指定查询中返回的单一对象 通过select属性指定用于多表查询的方法 使用格式:@Result(column=“”,property=“”,one=@One(select””))
@Many代替<collection>标签,用于指定查询中返回的集合对象 使用格式:@Result(column=“”,property=“”,many=@Many(select””))

结合实例进行分析:

一个用户拥有多个订单,一个订单是属于一个用户

image-20240514112738182

订单的表(uid对应用户编号)

image-20240514112812819

查用户

user类

 package com.example.mpdemo.entity;
 ​
 import com.baomidou.mybatisplus.annotation.IdType;
 import com.baomidou.mybatisplus.annotation.TableField;
 import com.baomidou.mybatisplus.annotation.TableId;
 import com.baomidou.mybatisplus.annotation.TableName;
 ​
 import java.util.List;
 ​
 @TableName("t_user")
 public class User {
     @TableId(type = IdType.AUTO)
     private int id;
     private String username;
     private String password;
     private String birthday;
 ​
 //  描述用户的所有订单 selcet id, username,password,orders from t_user;
     @TableField(exist = false)
     private List<Order> orders;
 ​
 //    alt + insert
 ​
     public int getId() {
         // 返回对象的id属性值
         return id;
     }
 ​
     public void setId(int id) {
         this.id = id;
     }
 ​
     public String getUsername() {
         return username;
     }
 ​
     public void setUsername(String username) {
         this.username = username;
     }
 ​
     public String getPassword() {
         return password;
     }
 ​
     public void setPassword(String password) {
         this.password = password;
     }
 ​
     public String getBirthday() {
         return birthday;
     }
 ​
     public void setBirthday(String birthday) {
         this.birthday = birthday;
     }
 ​
     public List<Order> getOrders() {
         return orders;
     }
 ​
     public void setOrders(List<Order> orders) {
         this.orders = orders;
     }
 ​
     @Override
     public String toString() {
         return "User{" +
                 "id=" + id +
                 ", username='" + username + '\'' +
                 ", password='" + password + '\'' +
                 ", birthday='" + birthday + '\'' +
                 ", orders=" + orders +
                 '}';
     }
 }

userMapper

 package com.example.mpdemo.mapper;
 ​
 import com.baomidou.mybatisplus.core.mapper.BaseMapper;
 import com.example.mpdemo.entity.User;
 import org.apache.ibatis.annotations.*;
 ​
 import java.util.List;
 ​
 //用于操作用户表,MyBaits会根据Mapper注解,动态实现UserMapper接口(实现类),动态代理技术
 //Spring会自动创建UserMapper接口实现类对应的实例
 @Mapper
 public interface UserMapper extends BaseMapper<User> {
 ​
 //   查询用户,根据用户id查询信息   select * from user where id =
     @Select("select * from t_user where id = #{id}")
     User selectById(int id);
 ​
 ​
 //   查询用户及其所有的订单
     @Select("select * from t_user")
     @Results(
             {
                     // 将数据库中的id列映射到User对象的id属性
                     @Result(column = "id",property = "id"),
                     // 将数据库中的username列映射到User对象的username属性
                     @Result(column = "username",property = "username"),
                     // 将数据库中的password列映射到User对象的password属性
                     @Result(column = "password",property = "password"),
                     // 将数据库中的birthday列映射到User对象的birthday属性
                     @Result(column = "birthday",property = "birthday"),
                     // 将数据库中的id列映射到User对象的orders属性,orders属性的类型为List
                     // 使用@Many注解指定关联的查询语句,查询当前用户的所有订单
                     @Result(column = "id",property = "orders",javaType = List.class,
                             many=@Many(select = "com.example.mpdemo.mapper.OrderMapper.selectByUid")
                     )
             }
     )
     // 查询所有用户及其所有订单的方法
     List<User> selectAllUserAndOrders();
 }

·``

测试(第一种:自动生成sql语句)

 //   查询用户及其所有的订单
     @Select("select * from t_user")
 ​
     // 查询所有用户及其所有订单的方法
     List<User> selectAllUserAndOrders();
 }
 ​
测试结果

image-20240514115601456

测试(第二种手动写映射)

 //   查询用户及其所有的订单
     @Select("select * from t_user")
     @Results(
             {
                     // 将数据库中的id列映射到User对象的id属性
                     @Result(column = "id",property = "id"),
                     // 将数据库中的username列映射到User对象的username属性
                     @Result(column = "username",property = "username"),
                     // 将数据库中的password列映射到User对象的password属性
                     @Result(column = "password",property = "password"),
                     // 将数据库中的birthday列映射到User对象的birthday属性
                     @Result(column = "birthday",property = "birthday"),
                     // 将数据库中的id列映射到User对象的orders属性,orders属性的类型为List(集合)
                     // 使用@Many注解指定关联的查询语句,查询当前用户的所有订单
                     @Result(column = "id",property = "orders",javaType = List.class,
                             many=@Many(select = "com.example.mpdemo.mapper.OrderMapper.selectByUid")
                     )
             }
     )
     // 查询所有用户及其所有订单的方法
     List<User> selectAllUserAndOrders();
 }

1.测试结果(查用户)

一个订单对应一个用户

image-20240516161407717

Order

 @TableName("t_order")
 public class Order {
     private int id;
     private String ordertime;
     private double total;
 ​
     @TableField(exist = false)
     private User user;
 ​
     public int getId() {
         return id;
     }
 ​
     public void setId(int id) {
         this.id = id;
     }
 ​
     public String getOrdertime() {
         return ordertime;
     }
 ​
     public void setOrdertime(String ordertime) {
         this.ordertime = ordertime;
     }
 ​
     public double getTotal() {
         return total;
     }
 ​
     public void setTotal(double total) {
         this.total = total;
     }
 ​
     public User getUser() {
         return user;
     }
 ​
     public void setUser(User user) {
         this.user = user;
     }
 ​
     @Override
     public String toString() {
         return "Orders{" +
                 "id=" + id +
                 ", ordertime='" + ordertime + '\'' +
                 ", total=" + total +
                 ", user=" + user +
                 '}';
     }
 }
 ​

创建OrderMapper

 @Mapper
 public interface OrderMapper extends BaseMapper<Order> {
 ​
     @Select("select * from t_order where uid = #{uid}")
     List<Order> selectByUid(int uid);
 ​
 //  查询所有的订单,同时查询订单的用户
     @Select("select * from t_order")
     @Results({
             @Result(column = "id",property = "id"),
             @Result(column = "ordertime",property = "ordertime"),
             @Result(column = "total",property = "total"),
             @Result(column = "uid",property = "user",javaType = User.class,
                 one=@One(select = "com.example.mpdemo.mapper.UserMapper.selectById")
             )
     })
     List<Order> selectAllOrdersAndUser();//查询订单所属用户
 }

2.测试结果查订单

image-20240516162439977

条件查询

usercontroller

 //  条件查询
     @GetMapping("/user/find")
     public List<User> findByCond(){
         QueryWrapper<User> queryWrapper = new QueryWrapper();
         queryWrapper.eq("username","zhangsan");
         return userMapper.selectList(queryWrapper);
     }

条件构造器

[条件构造器 | MyBatis-Plus (baomidou.com)]:

image-20240516163124569

三、分页查询

编写配置文件

 @Configuration
 public class MyBatisPlusConfig {
     @Bean
     public MybatisPlusInterceptor paginationInterceptor() {
         MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
         PaginationInnerInterceptor paginationInterceptor = new PaginationInnerInterceptor(DbType.MYSQL);//选择数据库
         interceptor.addInnerInterceptor(paginationInterceptor);//定义了一个分页的拦截器
         return interceptor;
 ​
     }
 }
 //可以调用selectPages方法

usercontroller

 //  分页查询 添加限制
     @GetMapping("/user/findByPage")
     public IPage findByPage(){
         //设置起始值及每页条数
         Page<User> page = new Page<>(0,2);//从第几条开始查,查几条
         IPage iPage = userMapper.selectPage(page,null);
         return iPage;
     }

测试:

  • 21
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值