【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 user;
INSERT 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);
}