【Mybatis-Plus笔记01】整合Springboot实现基础配置和增删改查案例

【一】Mybatis-Plus的简单介绍

【1】MP的特特性有哪些

MP是一个Mybatis的增强工具,在Mybatis的基础上只做增强不做改变,可以简化开发,高效效率。

特性如下:
(1)无侵入:只做增强不做改变,引入它不会对现有工程产生影响
(2)损耗小:启动即会自动注入基本 CURD,性能基本无损耗,直接面向对象操作
(3)强大的 CRUD 操作:内置通用 Mapper、通用 Service,仅仅通过少量配置即可实现单表大部分 CRUD 操作,更有强大的条件构造器,满足各类使用需求
(4)支持 Lambda 形式调用:通过 Lambda 表达式,方便的编写各类查询条件,无需再担心字段写错
(5)支持主键自动生成:支持多达 4 种主键策略(内含分布式唯一 ID 生成器 - Sequence),可自由配置,完美解决主键问题
(6)支持 ActiveRecord 模式:支持 ActiveRecord 形式调用,实体类只需继承 Model 类即可进行强大的 CRUD 操作
(7)支持自定义全局通用操作:支持全局通用方法注入( Write once, use anywhere )
(8)内置代码生成器:采用代码或者 Maven 插件可快速生成 Mapper 、 Model 、 Service 、 Controller 层代码,支持模板引擎,有更多的自定义配置
(9)内置分页插件:基于 MyBatis 物理分页,开发者无需关心具体操作,配置好插件之后,写分页等同于普通 List 查询
(10)分页插件支持多种数据库:支持 MySQL、MariaDB、Oracle、DB2、H2、HSQL、SQLite、Postgre、SQLServer 等多种数据库
(11)内置性能分析插件:可输出 SQL 语句以及其执行时间,建议开发测试时启用该功能,能快速揪出慢查询
(12) 内置全局拦截插件:提供全表 delete 、 update 操作智能分析阻断,也可自定义拦截规则,预防误操作

【2】MP的框架结构

在这里插入图片描述

【二】MP的使用案例

(1)准备开发环境

SpringBoot版本:2.6.3
Mysql版本:5.7
JDK版本:1.8
Mybatis-Plus版本:3.5.1

(2)添加pom依赖

<!--mybatis—plus-->
<dependency>
    <groupId>com.baomidou</groupId>
    <artifactId>mybatis-plus-boot-starter</artifactId>
    <version>3.5.1</version>
</dependency>
<!--lombok-->
<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <optional>true</optional>
</dependency>
<!--mysql-->
<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    <scope>runtime</scope>
</dependency>

(3)编写yml配置

spring:
  datasource:
    type: com.zaxxer.hikari.HikariDataSource
    driver-class-name: com.mysql.cj.jdbc.Driver # 现在的mysql驱动默认是8版本,所以选择带cj的
    url: jdbc:mysql://localhost:3306/mybatis_plus?characterEncoding=utf-8&useSSL=false
    username: root
    password:
mybatis-plus:
  configuration:
    # mybatis-plus自带的日志打印
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

(4)创建测试用的数据库

注意的是,这里的id主键将要用MP的雪花算法自动生成的id,所以类型是BIGINT

CREATE DATABASE `mybatis_plus`;
use `mybatis_plus`;

CREATE TABLE user
(
    id BIGINT(20) NOT NULL COMMENT '主键ID',
    name VARCHAR(30) NULL DEFAULT NULL COMMENT '姓名',
    age INT(11) NULL DEFAULT NULL COMMENT '年龄',
    email VARCHAR(50) NULL DEFAULT NULL COMMENT '邮箱',
    PRIMARY KEY (id)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

use userINSERT INTO user (id, name, age, email) VALUES
(1, 'Jone', 18, '[email protected]'),
(2, 'Jack', 20, '[email protected]'),
(3, 'Tom', 28, '[email protected]'),
(4, 'Sandy', 21, '[email protected]'),
(5, 'Billie', 24, '[email protected]');

(5)创建实体类

// @NoArgsConstructor // 无参构造
// @AllArgsConstructor // 有参构造
// @Getter // get方法
// @Setter // set方法
// @EqualsAndHashCode // equals和hashcoed方法
@Data // 包含以上,除了有参构造
@TableName("user")
public class User {
    
    
    private Long id;
    private String name;
    private Integer age;
    private String email;
}

(6)创建Mapper接口

(1)这里继承了BaseMapper,并且指定了泛型的类是User,BaseMapper里为我们提供了很多现成的方法
(2)在这里除了BaseMapper提供的方法,我们还可以自定义一些方法

@Repository
public interface UserMapper extends BaseMapper<User> {
    
    
    // 下面的方法都是自定义的方法了
    /**
     * 根据id查询用户信息为map集合
     * @MethodName: selectMapById
     * @Author: AllenSun
     * @Date: 2023/5/2 下午2:11
     */
    Map<String,Object> selectMapById(Long id);
}

(7)创建mapper.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.allen.springboot_mybatisplus.mapper.UserMapper">
    <!--Map<String,Object> selectMapById(Long id)-->
    <select id="selectMapById" resultType="map">
        select id,name,age,email from user where id=#{id}
    </select>
</mapper>

(8)创建Service接口

(1)继承了IService,这里提供了很多现成的方法

public interface UserService extends IService<User> {
    
    
}

(9)创建Service实现类

(1)除了实现自定义的UserService接口,还继承了ServiceImpl

@Service
public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements UserService {
    
    
}

(10)给springboot启动类加上注解@MapperScan

@SpringBootApplication
// 用来扫描mapper接口所在的包
@MapperScan("com.allen.springboot_mybatisplus.mapper")
public class SpringbootMybatisplusApplication {
    
    

    public static void main(String[] args) {
    
    
        SpringApplication.run(SpringbootMybatisplusApplication.class, args);
    }
}

(11)整体的结构

在这里插入图片描述

【三】MP封装的BaseMapper、IService、ServiceImpl里都有什么

【1】BaseMapper的CRUD方法

(1)通用 CRUD 封装BaseMapper
(opens new window)接口,为 Mybatis-Plus 启动时自动解析实体表关系映射转换为 Mybatis 内部对象注入容器
(2)泛型 T 为任意实体对象
(3)参数 Serializable 为任意类型主键 Mybatis-Plus 不推荐使用复合主键约定每一张表都有自己的唯一 id 主键
(4)对象 Wrapper 为 条件构造器

(1)Insert

// 插入一条记录
int insert(T entity);

在这里插入图片描述

(2)Delete

// 根据 entity 条件,删除记录
int delete(@Param(Constants.WRAPPER) Wrapper<T> wrapper);
// 删除(根据ID 批量删除)
int deleteBatchIds(@Param(Constants.COLLECTION) Collection<? extends Serializable> idList);
// 根据 ID 删除
int deleteById(Serializable id);
// 根据 columnMap 条件,删除记录
int deleteByMap(@Param(Constants.COLUMN_MAP) Map<String, Object> columnMap);

在这里插入图片描述

(3)Update

在调用updateById方法前,需要在T entity(对应的实体类)中的主键属性上加上@TableId注解。

// 根据 whereWrapper 条件,更新记录
int update(@Param(Constants.ENTITY) T updateEntity, @Param(Constants.WRAPPER) Wrapper<T> whereWrapper);
// 根据 ID 修改
int updateById(@Param(Constants.ENTITY) T entity);

在这里插入图片描述

(4)Select

// 根据 ID 查询
T selectById(Serializable id);
// 根据 entity 条件,查询一条记录
T selectOne(@Param(Constants.WRAPPER) Wrapper<T> queryWrapper);

// 查询(根据ID 批量查询)
List<T> selectBatchIds(@Param(Constants.COLLECTION) Collection<? extends Serializable> idList);
// 根据 entity 条件,查询全部记录
List<T> selectList(@Param(Constants.WRAPPER) Wrapper<T> queryWrapper);
// 查询(根据 columnMap 条件)
List<T> selectByMap(@Param(Constants.COLUMN_MAP) Map<String, Object> columnMap);
// 根据 Wrapper 条件,查询全部记录
List<Map<String, Object>> selectMaps(@Param(Constants.WRAPPER) Wrapper<T> queryWrapper);
// 根据 Wrapper 条件,查询全部记录。注意: 只返回第一个字段的值
List<Object> selectObjs(@Param(Constants.WRAPPER) Wrapper<T> queryWrapper);

// 根据 entity 条件,查询全部记录(并翻页)
IPage<T> selectPage(IPage<T> page, @Param(Constants.WRAPPER) Wrapper<T> queryWrapper);
// 根据 Wrapper 条件,查询全部记录(并翻页)
IPage<Map<String, Object>> selectMapsPage(IPage<T> page, @Param(Constants.WRAPPER) Wrapper<T> queryWrapper);
// 根据 Wrapper 条件,查询总记录数
Integer selectCount(@Param(Constants.WRAPPER) Wrapper<T> queryWrapper);

在这里插入图片描述

【2】IService的CRUD方法

(1)通用 Service CRUD 封装IService
(opens new window)接口,进一步封装 CRUD 采用 get 查询单行 remove 删除 list 查询集合 page 分页 前缀命名方式区分 Mapper 层避免混淆,
(2)泛型 T 为任意实体对象
(3)建议如果存在自定义通用 Service 方法的可能,请创建自己的 IBaseService 继承 Mybatis-Plus 提供的基类
(4)对象 Wrapper 为 条件构造器

(1)Save

// 插入一条记录(选择字段,策略插入)
boolean save(T entity);
// 插入(批量)
boolean saveBatch(Collection<T> entityList);
// 插入(批量)
boolean saveBatch(Collection<T> entityList, int batchSize);

在这里插入图片描述

(2)SaveOrUpdate

// TableId 注解存在更新记录,否插入一条记录
boolean saveOrUpdate(T entity);
// 根据updateWrapper尝试更新,否继续执行saveOrUpdate(T)方法
boolean saveOrUpdate(T entity, Wrapper<T> updateWrapper);
// 批量修改插入
boolean saveOrUpdateBatch(Collection<T> entityList);
// 批量修改插入
boolean saveOrUpdateBatch(Collection<T> entityList, int batchSize);

在这里插入图片描述

(3)Remove

// 根据 entity 条件,删除记录
boolean remove(Wrapper<T> queryWrapper);
// 根据 ID 删除
boolean removeById(Serializable id);
// 根据 columnMap 条件,删除记录
boolean removeByMap(Map<String, Object> columnMap);
// 删除(根据ID 批量删除)
boolean removeByIds(Collection<? extends Serializable> idList);

在这里插入图片描述

(4)Update

// 根据 UpdateWrapper 条件,更新记录 需要设置sqlset
boolean update(Wrapper<T> updateWrapper);
// 根据 whereWrapper 条件,更新记录
boolean update(T updateEntity, Wrapper<T> whereWrapper);
// 根据 ID 选择修改
boolean updateById(T entity);
// 根据ID 批量更新
boolean updateBatchById(Collection<T> entityList);
// 根据ID 批量更新
boolean updateBatchById(Collection<T> entityList, int batchSize);

在这里插入图片描述

(5)Get

// 根据 ID 查询
T getById(Serializable id);
// 根据 Wrapper,查询一条记录。结果集,如果是多个会抛出异常,随机取一条加上限制条件 wrapper.last("LIMIT 1")
T getOne(Wrapper<T> queryWrapper);
// 根据 Wrapper,查询一条记录
T getOne(Wrapper<T> queryWrapper, boolean throwEx);
// 根据 Wrapper,查询一条记录
Map<String, Object> getMap(Wrapper<T> queryWrapper);
// 根据 Wrapper,查询一条记录
<V> V getObj(Wrapper<T> queryWrapper, Function<? super Object, V> mapper);

在这里插入图片描述

(6)List

// 查询所有
List<T> list();
// 查询列表
List<T> list(Wrapper<T> queryWrapper);
// 查询(根据ID 批量查询)
Collection<T> listByIds(Collection<? extends Serializable> idList);
// 查询(根据 columnMap 条件)
Collection<T> listByMap(Map<String, Object> columnMap);
// 查询所有列表
List<Map<String, Object>> listMaps();
// 查询列表
List<Map<String, Object>> listMaps(Wrapper<T> queryWrapper);
// 查询全部记录
List<Object> listObjs();
// 查询全部记录
<V> List<V> listObjs(Function<? super Object, V> mapper);
// 根据 Wrapper 条件,查询全部记录
List<Object> listObjs(Wrapper<T> queryWrapper);
// 根据 Wrapper 条件,查询全部记录
<V> List<V> listObjs(Wrapper<T> queryWrapper, Function<? super Object, V> mapper);

在这里插入图片描述

(7)Page

// 无条件分页查询
IPage<T> page(IPage<T> page);
// 条件分页查询
IPage<T> page(IPage<T> page, Wrapper<T> queryWrapper);
// 无条件分页查询
IPage<Map<String, Object>> pageMaps(IPage<T> page);
// 条件分页查询
IPage<Map<String, Object>> pageMaps(IPage<T> page, Wrapper<T> queryWrapper);

在这里插入图片描述

(8)Count

// 查询总记录数
int count();
// 根据 Wrapper 条件,查询总记录数
int count(Wrapper<T> queryWrapper);

在这里插入图片描述

【四】MQ的测试案例

【1】Mapper的方法测试

package com.allen.springboot_mybatisplus;

import com.allen.springboot_mybatisplus.mapper.UserMapper;
import com.allen.springboot_mybatisplus.pojo.User;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
 * @ClassName: MybatisPlusTest
 * @Author: AllenSun
 * @Date: 2023/5/1 下午11:37
 */
@SpringBootTest
public class MybatisPlusTest {
    
    

    @Autowired
    private UserMapper userMapper;

    @Test
    public void selectList() {
    
    
        // 通过条件构造器查询一个List集合,如果没有条件,就可以设置null为参数
        List<User> list = userMapper.selectList(null);
        list.forEach(System.out::println);
    }

    @Test
    public void selectById() {
    
    
        User user = userMapper.selectById(1L);
        System.out.println("查询结果:"+user.toString());
    }

    @Test
    public void selectByMap() {
    
    
        Map<String,Object> conditionMap = new HashMap<>();
        conditionMap.put("name","Tom");
        conditionMap.put("age",28);
        List<User> list = userMapper.selectByMap(conditionMap);
        list.forEach(System.out::println);
    }

    @Test
    public void selectBatchIds() {
    
    
        List<Long> list = Arrays.asList(1L,2L,3L);
        // SELECT id,name,age,email FROM user WHERE id IN ( ? , ? , ? )
        List<User> result = userMapper.selectBatchIds(list);
        result.forEach(System.out::println);
    }


    @Test
    public void selfSelectMapById() {
    
    
        Map<String,Object> map = userMapper.selectMapById(1L);
        System.out.println("查询结果:"+map);
    }

    @Test
    public void insert() {
    
    
        User user = new User();
        user.setName("Allen");
        user.setAge(23);
        user.setEmail("[email protected]");
        int insertResult = userMapper.insert(user);
        System.out.println("插入结果:"+insertResult);
        System.out.println("插入id:"+user.getId());
    }

    // 根据参数的不同有很多个方法
    @Test
    public void deleteById() {
    
    
        // 通过id删除
        int deleteResult = userMapper.deleteById(1653072204772282370L);
        System.out.println("删除结果:"+deleteResult);
    }

    @Test
    public void deleteByMap() {
    
    
        Map<String,Object> conditionMap = new HashMap<>();
        conditionMap.put("name","Allen");
        conditionMap.put("age",23);
        int deleteResult = userMapper.deleteByMap(conditionMap);
        System.out.println("删除结果:"+deleteResult);

    }

    @Test
    public void deleteBatchIds() {
    
    
        List<Long> list = Arrays.asList(1L,2L,3L);
        int deleteResult = userMapper.deleteBatchIds(list);
        System.out.println("删除结果:"+deleteResult);
    }

    @Test
    public void updateById() {
    
    
        User user = new User();
        user.setId(4L);
        user.setName("Sandy Update");
        user.setAge(12);
        user.setEmail("[email protected]");
        // UPDATE user SET name=?, age=?, email=? WHERE id=?
        int updateResult = userMapper.updateById(user);
        System.out.println("修改结果:"+updateResult);
    }

    @Test
    public void update() {
    
    

    }
}

【2】Service的方法测试

package com.allen.springboot_mybatisplus;

import com.allen.springboot_mybatisplus.pojo.User;
import com.allen.springboot_mybatisplus.service.UserService;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

import java.util.ArrayList;
import java.util.List;

/**
 * @ClassName: MybatisPlusServieTest
 * @Author: AllenSun
 * @Date: 2023/5/2 下午2:56
 */
@SpringBootTest
public class MybatisPlusServiceTest {
    
    

    @Autowired
    private UserService userService;

    @Test
    public void getCount() {
    
    
        Long count = userService.count();
        System.out.println("查询总数为:"+count);
    }

    @Test
    public void saveBatch() {
    
    
        List<User> list = new ArrayList<>();
        for (int i = 0; i < 10; i++) {
    
    
            User user = new User();
            user.setName("Allen"+i);
            user.setAge(20+i);
            list.add(user);
        }
        boolean result = userService.saveBatch(list);
        System.out.println("插入结果:"+result);
    }
}

【3】Wrapper的方法测试

package com.allen.springboot_mybatisplus;

import com.allen.springboot_mybatisplus.mapper.UserMapper;
import com.allen.springboot_mybatisplus.pojo.User;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import org.junit.jupiter.api.Test;
import org.junit.platform.commons.util.StringUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

import java.util.List;

/**
 * @ClassName: MybatisPlusWrapperTest
 * @Author: AllenSun
 * @Date: 2023/5/2 下午3:37
 */
@SpringBootTest
public class MybatisPlusWrapperTest {
    
    

    @Autowired
    private UserMapper userMapper;

    @Test
    public void selectByWrapper() {
    
    
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.like("name","Allen")
                .between("age",20,30)
                .isNotNull("email");
        // SELECT id,name,age,email FROM user WHERE (name LIKE ? AND age BETWEEN ? AND ? AND email IS NOT NULL)
        List<User> list = userMapper.selectList(queryWrapper);
        list.forEach(System.out::println);
    }

    @Test
    public void selectByOrder() {
    
    
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.orderByAsc("age").orderByDesc("id");
        // SELECT id,name,age,email FROM user ORDER BY age ASC,id DESC
        List<User> list = userMapper.selectList(queryWrapper);
        list.forEach(System.out::println);
    }

    @Test
    public void selectByField() {
    
    
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.select("name","age");
        // SELECT name,age FROM user
        List<User> list = userMapper.selectList(queryWrapper);
        list.forEach(System.out::println);
    }

    @Test
    public void selectByIn() {
    
    
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.inSql("id","select id from user where id<=100");
        // SELECT id,name,age,email FROM user WHERE (id IN (select id from user where id<=100))
        List<User> list = userMapper.selectList(queryWrapper);
        list.forEach(System.out::println);
    }

    @Test
    public void selectByCondition() {
    
    

        String name="";
        Integer ageBegin = 20;
        Integer ageEnd = 30;

        QueryWrapper<User> queryWrapper = new QueryWrapper<>();

        if (StringUtils.isNotBlank(name)){
    
    
            queryWrapper.like("name",name);
        }
        if (ageBegin!=null) {
    
    
            queryWrapper.ge("age",ageBegin);
        }
        if (ageEnd!=null) {
    
    
            queryWrapper.le("age",ageEnd);
        }

        // SELECT id,name,age,email FROM user WHERE (age >= ? AND age <= ?)
        List<User> list = userMapper.selectList(queryWrapper);
        list.forEach(System.out::println);
    }

    @Test
    public void selectByCondition2() {
    
    

        String name="";
        Integer ageBegin = 20;
        Integer ageEnd = 30;

        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.like(StringUtils.isNotBlank(name),"name",name)
                .ge(ageBegin!=null,"age",ageBegin)
                .le(ageEnd!=null,"age",ageEnd);

        // SELECT id,name,age,email FROM user WHERE (age >= ? AND age <= ?)
        List<User> list = userMapper.selectList(queryWrapper);
        list.forEach(System.out::println);
    }

    @Test
    public void updateByOr() {
    
    
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.gt("age",20)
                .like("name","Allen")
                .or()
                .isNull("email");
        User user = new User();
        user.setName("小明");
        user.setEmail("[email protected]");
        // UPDATE user SET name=?, email=? WHERE (age > ? AND name LIKE ? OR email IS NULL)
        int result = userMapper.update(user,queryWrapper);
        System.out.println("修改结果:"+result);
    }
}

【五】常用注解

【1】@TableName

(1)MP是如何确定要操作的是哪张表的?
在使用MyBatis-Plus实现基本的CRUD时,我们并没有指定要操作的表,只是在Mapper接口继承BaseMapper时,设置了泛型User,而操作的表为user表。

所以,MyBatis-Plus在确定操作的表时,由BaseMapper的泛型决定,即实体类型决定,且默认操作的表名和实体类型的类名一致

(2)如果实体类类型的类名和要操作的表名不一致会怎么样?
如果我们将表user更名为t_user,测试查询功能,程序抛出异常,Table ‘mybatis_plus.user’ doesn’t exist,因为现在的表名为t_user,而默认操作的表名和实体类型的类名一致,即user表。

(3)表名和实体类名不一致怎么办?
在实体类类型上添加@TableName(“t_user”),标识实体类对应的表,即可成功执行SQL语句。

这样就可以把User实体类和表t_user关联起来了
在这里插入图片描述
(4)这种表的统一前缀还可以通过全局配置来解决

【2】@TableId

(1)MP是怎么确认主键?
MyBatis-Plus在实现CRUD时,会默认将id作为主键列,并在插入数据时,默认基于雪花算法的策略生成id

(2)如果实体类和表中的主键不是id,而是其他字段,例如uid,MyBatis-Plus会自动识别uid为主键列吗?
我们实体类中的属性id改为uid,将表中的字段id也改为uid,测试添加功能。程序抛出异常,Field ‘uid’ doesn’t have a default value,说明MyBatis-Plus没有将uid作为主键赋值

(3)通过@TableId结局实体类中主键值不是id
在实体类中uid属性上通过@TableId将其标识为主键,即可成功执行SQL语句
在这里插入图片描述在这里插入图片描述

(4)解决表中主键值不是id
如果实体类中主键对应的属性为id,而表中表示主键的字段为uid,此时若只在属性id上添加注解
@TableId,则抛出异常Unknown column ‘id’ in ‘field list’,即MyBatis-Plus仍然会将id作为表的主键操作,而表中表示主键的是字段uid。

此时需要通过@TableId注解的value属性,指定表中的主键字段,@TableId(“uid”)或
@TableId(value=“uid”)

(5)MP是怎么生成主键的值?
默认是雪花算法

(6)如何修改默认的生成主键策略?
修改@TableId的type属性,改成AUTO,就是自增ID了,使用数据库的自增策略,注意,该类型请确保数据库设置了id自增,否则无效
在这里插入图片描述

【3】@TableField

(1)MP可以自动完成实体类的字段和表的值对应
MyBatis-Plus在执行SQL语句时,要保证实体类中的属性名和表中的字段名一致

(2)如果实体类中的属性名和字段名不一致的情况,会出现什么问题呢?
若实体类中的属性使用的是驼峰命名风格,而表中的字段使用的是下划线。命名风格例如实体类属性userName,表中字段user_name,此时MyBatis-Plus会自动将下划线命名风格转化为驼峰命名风格,相当于在MyBatis中配置。

(3)如果是名字完全对不上怎么办?
例如实体类属性name,表中字段username。此时需要在实体类属性上使用@TableField(“username”)设置属性所对应的字段名

【4】@TableLogic

(1)什么是逻辑删除?
1-物理删除:真实删除,将对应数据从数据库中删除,之后查询不到此条被删除的数据
2-逻辑删除:假删除,将对应数据中代表是否被删除字段的状态修改为“被删除状态”,之后在数据库中仍旧能看到此条数据记录
3-使用场景:可以进行数据恢复

(2)如何实现逻辑删除
1-数据库中创建逻辑删除状态列,设置默认值为0
在这里插入图片描述

2-实体类中添加逻辑删除属性
在这里插入图片描述

3-测试
测试删除功能,真正执行的是修改
UPDATE t_user SET is_deleted=1 WHERE id=? AND is_deleted=0
测试查询功能,被逻辑删除的数据默认不会被查询
SELECT id,username AS name,age,email,is_deleted FROM t_user WHERE is_deleted=0

【六】条件构造器和常用接口

【1】QueryWrapper

(1)组装查询条件

@Test 
public void test01(){
    
     
	//查询用户名包含a,年龄在20到30之间,并且邮箱不为null的用户信息 
	//SELECT id,username AS name,age,email,is_deleted FROM t_user WHERE is_deleted=0 AND (username LIKE ? AND age BETWEEN ? AND ? AND email IS NOT NULL) 
	QueryWrapper<User> queryWrapper = new QueryWrapper<>();
	queryWrapper.like("username", "a") .between("age", 20, 30) .isNotNull("email"); 
	List<User> list = userMapper.selectList(queryWrapper);
	list.forEach(System.out::println); 
}

(2)组装排序条件

@Test 
public void test02(){
    
     
	//按年龄降序查询用户,如果年龄相同则按id升序排列 
	//SELECT id,username AS name,age,email,is_deleted FROM t_user WHERE is_deleted=0 ORDER BY age DESC,id ASC 
	QueryWrapper<User> queryWrapper = new QueryWrapper<>(); 
	queryWrapper .orderByDesc("age") .orderByAsc("id"); 
	List<User> users = userMapper.selectList(queryWrapper); 
	users.forEach(System.out::println); 
}

(3)组装删除条件

@Test 
public void test03(){
    
     
	//删除email为空的用户 
	//DELETE FROM t_user WHERE (email IS NULL) 
	QueryWrapper<User> queryWrapper = new QueryWrapper<>(); 
	queryWrapper.isNull("email"); 
	//条件构造器也可以构建删除语句的条件 
	int result = userMapper.delete(queryWrapper); 
	System.out.println("受影响的行数:" + result); 
}

(4)组装select子句

@Test 
public void test05() {
    
     
	//查询用户信息的username和age字段 
	//SELECT username,age FROM t_user 
	QueryWrapper<User> queryWrapper = new QueryWrapper<>(); 
	queryWrapper.select("username", "age"); 
	//selectMaps()返回Map集合列表,通常配合select()使用,避免User对象中没有被查询到的列值 为null
	List<Map<String, Object>> maps = userMapper.selectMaps(queryWrapper); 
	maps.forEach(System.out::println); 
}

(5)实现子查询

@Test 
public void test06() {
    
     
	//查询id小于等于3的用户信息 
	//SELECT id,username AS name,age,email,is_deleted FROM t_user WHERE (id IN (select id from t_user where id <= 3)) 
	QueryWrapper<User> queryWrapper = new QueryWrapper<>(); 
	queryWrapper.inSql("id", "select id from t_user where id <= 3"); 
	//selectObjs的使用场景:只返回一列 
	List<Object> objects = userMapper.selectObjs(queryWrapper); 
	objects.forEach(System.out::println); 
}

【2】UpdateWrapper

【3】condition

在真正开发的过程中,组装条件是常见的功能,而这些条件数据来源于用户输入,是可选的,因此我们在组装这些条件时,必须先判断用户是否选择了这些条件,若选择则需要组装该条件,若没有选择则一定不能组装,以免影响SQL执行的结果。

(1)方式一

@Test 
public void test08() {
    
     
	//定义查询条件,有可能为null(用户未输入或未选择) 
	String username = null; 
	Integer ageBegin = 10; 
	Integer ageEnd = 24; 
	QueryWrapper<User> queryWrapper = new QueryWrapper<>(); 
	//StringUtils.isNotBlank()判断某字符串是否不为空且长度不为0且不由空白符(whitespace) 构成 
	if(StringUtils.isNotBlank(username)){
    
     
		queryWrapper.like("username","a"); 
	}
	if(ageBegin != null){
    
     
		queryWrapper.ge("age", ageBegin); 
	}
	if(ageEnd != null){
    
     
		queryWrapper.le("age", ageEnd); 
	}
	//SELECT id,username AS name,age,email,is_deleted FROM t_user WHERE (age >= ? AND age <= ?) 
	List<User> users = userMapper.selectList(queryWrapper); 
	users.forEach(System.out::println); 
}

(2)方式二

上面的实现方案没有问题,但是代码比较复杂,我们可以使用带condition参数的重载方法构建查询条件,简化代码的编写

@Test 
public void test08UseCondition() {
    
     
	//定义查询条件,有可能为null(用户未输入或未选择) 
	String username = null; 
	Integer ageBegin = 10; 
	Integer ageEnd = 24; 
	QueryWrapper<User> queryWrapper = new QueryWrapper<>(); 
	//StringUtils.isNotBlank()判断某字符串是否不为空且长度不为0且不由空白符(whitespace) 构成 
	queryWrapper 
		.like(StringUtils.isNotBlank(username), "username", "a") 
		.ge(ageBegin != null, "age", ageBegin) 
		.le(ageEnd != null, "age", ageEnd); 
	//SELECT id,username AS name,age,email,is_deleted FROM t_user WHERE (age >= ? AND age <= ?) 
	List<User> users = userMapper.selectList(queryWrapper); 
	users.forEach(System.out::println); 
}

猜你喜欢

转载自blog.csdn.net/weixin_44823875/article/details/130465046