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

news2025/1/13 10:19:17

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

  • 【一】Mybatis-Plus的简单介绍
    • 【1】MP的特特性有哪些
    • 【2】MP的框架结构
  • 【二】MP的使用案例
    • (1)准备开发环境
    • (2)添加pom依赖
    • (3)编写yml配置
    • (4)创建测试用的数据库
    • (5)创建实体类
    • (6)创建Mapper接口
    • (7)创建mapper.xml映射配置
    • (8)创建Service接口
    • (9)创建Service实现类
    • (10)给springboot启动类加上注解@MapperScan
    • (11)整体的结构
  • 【三】MP封装的BaseMapper、IService、ServiceImpl里都有什么
    • 【1】BaseMapper的CRUD方法
      • (1)Insert
      • (2)Delete
      • (3)Update
      • (4)Select
    • 【2】IService的CRUD方法
      • (1)Save
      • (2)SaveOrUpdate
      • (3)Remove
      • (4)Update
      • (5)Get
      • (6)List
      • (7)Page
      • (8)Count
  • 【四】MQ的测试案例
    • 【1】Mapper的方法测试
    • 【2】Service的方法测试
    • 【3】Wrapper的方法测试
  • 【五】常用注解
    • 【1】@TableName
    • 【2】@TableId
    • 【3】@TableField
    • 【4】@TableLogic
  • 【六】条件构造器和常用接口
    • 【1】QueryWrapper
      • (1)组装查询条件
      • (2)组装排序条件
      • (3)组装删除条件
      • (4)组装select子句
      • (5)实现子查询
    • 【2】UpdateWrapper
    • 【3】condition
      • (1)方式一
      • (2)方式二

【一】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, 'test1@baomidou.com'),
(2, 'Jack', 20, 'test2@baomidou.com'),
(3, 'Tom', 28, 'test3@baomidou.com'),
(4, 'Sandy', 21, 'test4@baomidou.com'),
(5, 'Billie', 24, 'test5@baomidou.com');

(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("allen@qq.com");
        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("test44@baomidou.com");
        // 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("test@qq.com");
        // 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); 
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/494907.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于SpringBoot+Vue实现的体检录入系统

【简介】 本体检信息录入系统采用前端&#xff1a;vue&#xff1b;后端&#xff1a;springbootmybatis-plusredismysql技术架构开发&#xff0c;前后端分离&#xff0c;容易上手。除了基本的体检结果查询、录入及导出外&#xff0c;在录入中还能对录入信息进行智能计算。 【功…

LeetCode:20. 有效的括号

20. 有效的括号 1&#xff09;题目2&#xff09;思路3&#xff09;代码4&#xff09;结果 1&#xff09;题目 给定一个只包括 (&#xff0c;)&#xff0c;{&#xff0c;}&#xff0c;[&#xff0c;] 的字符串 s &#xff0c;判断字符串是否有效。 有效字符串需满足&#xff1…

Multi-Head self-Attention结构细节

先验知识&#xff1a; Self-Attention结构细节及计算过程https://blog.csdn.net/weixin_54039182/article/details/130515594?csdn_share_tail%7B%22type%22%3A%22blog%22%2C%22rType%22%3A%22article%22%2C%22rId%22%3A%22130515594%22%2C%22source%22%3A%22weixin_54039182…

分布式夺命12连问

分布式理论 1. 说说CAP原则&#xff1f; CAP原则又称CAP定理&#xff0c;指的是在一个分布式系统中&#xff0c;Consistency&#xff08;一致性&#xff09;、 Availability&#xff08;可用性&#xff09;、Partition tolerance&#xff08;分区容错性&#xff09;这3个基本…

Zotero 抓取知网文献失败解决办法梳理(针对问题:更新Translator无效,更新茉莉花插件无效,卸载Zotero无效,使用学校VPN访问知网)

Zotero 抓取知网文献失败解决办法梳理&#xff08;针对问题&#xff1a;更新Translator无效&#xff0c;更新茉莉花插件无效&#xff0c;卸载Zotero无效&#xff0c;使用学校VPN访问知网&#xff09; 问题背景解决方案一解决方案二解决方案三 说在前面&#xff1a;解决方案一和…

idea 远程debug阿里云服务器springboot项目

文章目录 前言意见和建议技术要点小试牛刀通信端口放行Idea增加remote启动项服务端JAR增加参数并启动本地项目启动远程debugpostman测试debug 前言 在实际的生产中不免会出现系统问题&#xff0c;有的在测试环境发现&#xff0c;有的在预发布环境发现&#xff0c;更有甚者在生…

播放卡顿分析

看下这个M3U8请求&#xff0c;时间间隔超过duration的时长。ts的duration是11S M3U8内容更新慢&#xff1f;

JavaScript:二叉树(前序遍历,中序遍历,后序遍历,递归法,统一迭代法)

文章目录 二叉树递归法迭代法 144. 二叉树的前序遍历 - 力扣&#xff08;LeetCode&#xff09;二叉树的递归遍历递归法作图分析代码和思路分析 二叉树的迭代遍历前序遍历迭代分析代码及思路分析 94. 二叉树的中序遍历递归法作图举例递归流程 迭代法代码 145. 二叉树的后序遍历 …

虹科分享|便携式数据包捕获解决方案的发展

有人说&#xff0c;未来就在眼前。如果我们看看过去十年中开发出的物联网技术&#xff0c;我们的确没发反驳他们。21世纪的技术繁荣改变了我们的生活&#xff0c;和彼此之间的交流方式。 比如说我们正在研究的MAREA项目&#xff0c;我们甚至可以说我们正在见证历史。 这是一个…

晚唐诗人杜荀鹤及其十首古诗赏析

一、关于出身的传说 他出身寒微。曾数次赴长安应考&#xff0c;不第还山。相传他是杜牧出妾之子。他诗语言通俗、风格清新&#xff0c;后人称“杜荀鹤体”。他就是晚唐诗人杜荀鹤。 据说&#xff0c;杜牧在会昌末年任池州刺史时&#xff0c;妾程氏有孕&#xff0c;为杜妻所逐&…

翻译|英译汉|汉译英|11:30-11:50+8:40-10:00

英译汉&#xff1a;逐句翻译、注意用词、确保大体通顺。 目录 一、解题技巧 &#xff08;一&#xff09;词语翻译 1. 词的选用 2. 词性转换 &#xff08;1&#xff09;英译汉中的词性转换 &#xff08;2&#xff09;汉译英中的词性转换 3. 增词法 4. 减词法 &#xff…

白嫖党的福音,这5款实用的黑科技软件,干货满满,真是相见恨晚

赶紧上车&#xff01;一般人都不知道的5款超实用软件&#xff0c;个个都是黑科技工具&#xff01;用了简直效率翻倍&#xff0c;不允许你不知道&#xff01;&#xff01;&#xff01;先收藏&#xff0c;以备不时之需。 1.quicklook 作为windows最强的文件预览器&#xff0c;q…

IGBT基础知识

1. 什么是IGBT&#xff1f; IGBT&#xff0c;绝缘栅双极型晶体管&#xff0c;是由&#xff08;BJT&#xff09;双极型三极管和绝缘栅型场效应管&#xff08;MOS&#xff09;组成的复合全控型电压驱动式功率半导体器件, 兼有&#xff08;MOSFET&#xff09;金氧半场效晶体管的高…

【虚拟机数据恢复】Vmware ESXI虚拟机数据恢复案例

虚拟机数据恢复环境&#xff1a; ESXI上共有数十台虚拟机&#xff0c;EXSI连接一台HP EVA存储&#xff0c;所有虚拟机都存放在该EVA存储上。 其中一台虚拟机是数年前从物理机迁移过来的&#xff0c;其上部署了一个SQL SERVER数据库&#xff0c;该数据库存放了最近几年的数据。 …

【电子学会】2023年03月图形化四级 -- 判断亲和数

判断亲和数 对于正整数a和b&#xff08;a>b&#xff09;,如果a除以b的余数为0&#xff0c;那么b是a的因数&#xff0c;例如6的因数为1,2,3,6。 亲和数&#xff0c;指两个正整数&#xff0c;彼此的全部因数之和&#xff08;这里的因数不包括整数自己&#xff09;与另一方相…

Qt_C++读写t5557卡复制HID卡源码

T5557卡是美国Atmel公司生产的多功能非接触式射频卡芯片&#xff0c;属于125KHz的低频卡&#xff0c;在国内有广大的应用市场。该芯片共有330bit(比特)的EPROM(分布为10个区块, 每个区块33bit)。0页的块0是被保留用于设置T5557操作模式的参数配置块。第0页第7块可以作用户数据块…

Qt5.9学习笔记-事件(三) 多线程和事件处理

⭐️我叫忆_恒心&#xff0c;一名喜欢书写博客的在读研究生&#x1f468;‍&#x1f393;。 如果觉得本文能帮到您&#xff0c;麻烦点个赞&#x1f44d;呗&#xff01; 近期会不断在专栏里进行更新讲解博客~~~ 有什么问题的小伙伴 欢迎留言提问欧&#xff0c;喜欢的小伙伴给个三…

OpenCV中的图像处理3.4-3.6(四)平滑化、形态学、图像梯度

目录 3.4 平滑化图像目标二维卷积 ( 图像滤波 )图像模糊(图像平滑)其他资源 3.5 形态学转换目标理论结构化元素 3.6 图像梯度目标理论代码一个重要的问题! 翻译及二次校对&#xff1a;cvtutorials.com 编辑者&#xff1a;廿瓶鲸&#xff08;和鲸社区Siby团队成员&#xff09; …

用 Bitmap 实现亿级海量数据统计

在移动应用的业务场景中&#xff0c;我们需要保存这样的信息&#xff1a;一个 key 关联了一个数据集合。 常见的场景如下&#xff1a; 给一个 userId &#xff0c;判断用户登陆状态&#xff1b; 显示用户某个月的签到次数和首次签到时间&#xff1b; 两亿用户最近 7 天的签到…

【C++:模块3-------数组】

C&#xff1a;模块3-------数组 数组概念&#xff1a;数组特点&#xff1a;数组访问方法&#xff1a; 一维数组&#xff1a;1.1三种定义形式&#xff1a;&#xff08;和C语言中一样&#xff09;1.2一维数组数组名的作用&#xff1a;&#xff08;1&#xff09; 统计数组在内存中…