第一章:ActiveRecord

1.1 概述(摘自官网)

  • ActiveRecord(简称AR)一直广受动态语言( PHP 、 Ruby 等)的喜爱,而 Java 作为准静态语言,对于
    ActiveRecord 往往只能感叹其优雅,所以我们也在 AR 道路上进行了一定的探索,喜欢大家能够喜欢。

什么是ActiveRecord? ActiveRecord也属于ORM(对象关系映射)层,由Rails最早提出,遵循标准的ORM模型:表映射到记录,记
录映射到对象,字段映射到对象属性。配合遵循的命名和配置惯例,能够很大程度的快速实现模型的操作,而
且简洁易懂。

ActiveRecord的主要思想是:

  • 每一个数据库表对应创建一个类,类的每一个对象实例对应于数据库中表的一行记录;通常表的每个字段在类中都有相应的Field;
  • ActiveRecord同时负责把自己持久化,在ActiveRecord中封装了对数据库的访问,即CURD;
  • ActiveRecord是一种领域模型(Domain Model),封装了部分业务逻辑;

1.2 开启AR之旅

  • 在MP中,开启AR非常简单,只需要实体对象继承Model即可,同时还需要重写pkVal方法,并返回主键。
  • 实体对象:
  1. package com.example.mp.domain;
  2. import com.baomidou.mybatisplus.annotation.IdType;
  3. import com.baomidou.mybatisplus.annotation.TableField;
  4. import com.baomidou.mybatisplus.annotation.TableId;
  5. import com.baomidou.mybatisplus.annotation.TableName;
  6. import com.baomidou.mybatisplus.extension.activerecord.Model;
  7. import lombok.AllArgsConstructor;
  8. import lombok.Data;
  9. import lombok.NoArgsConstructor;
  10. import lombok.ToString;
  11. import lombok.experimental.Accessors;
  12. import java.io.Serializable;
  13. /**
  14. * @author 许大仙
  15. * @version 1.0
  16. * @since 2021-06-17 21:01
  17. */
  18. @Data
  19. @NoArgsConstructor
  20. @AllArgsConstructor
  21. @TableName("tb_user")
  22. @Accessors(chain = true)
  23. @ToString
  24. public class User extends Model<User> {
  25. @TableId(type= IdType.AUTO)
  26. private Long id;
  27. @TableField("user_name") //解决字段名不一致
  28. private String username;
  29. private String password;
  30. private String name;
  31. private Integer age;
  32. private String email;
  33. @Override
  34. public Serializable pkVal() {
  35. return id;
  36. }
  37. }
  • UserMapper.java
  1. package com.example.mp.mapper;
  2. import com.baomidou.mybatisplus.core.mapper.BaseMapper;
  3. import com.example.mp.domain.User;
  4. import org.apache.ibatis.annotations.Mapper;
  5. /**
  6. * @author 许大仙
  7. * @version 1.0
  8. * @since 2021-06-17 21:10
  9. */
  10. @Mapper
  11. public interface UserMapper extends BaseMapper<User> {
  12. }

注意:UserMapper不能删除。

1.3 根据主键查询

  • 示例:
  1. package com.example.mp;
  2. import com.example.mp.domain.User;
  3. import org.junit.jupiter.api.Test;
  4. import org.springframework.boot.test.context.SpringBootTest;
  5. /**
  6. * @author 许大仙
  7. * @version 1.0
  8. * @since 2021-06-19 18:47
  9. */
  10. @SpringBootTest
  11. public class TestActiveRecord {
  12. @Test
  13. public void testSelectById() {
  14. User user = new User().setId(1L).selectById();
  15. System.out.println("user = " + user);
  16. }
  17. }

1.4 新增数据

  • 示例:
  1. package com.example.mp;
  2. import com.example.mp.domain.User;
  3. import org.junit.jupiter.api.Test;
  4. import org.springframework.boot.test.context.SpringBootTest;
  5. /**
  6. * @author 许大仙
  7. * @version 1.0
  8. * @since 2021-06-19 18:47
  9. */
  10. @SpringBootTest
  11. public class TestActiveRecord {
  12. @Test
  13. public void testInsert(){
  14. boolean flag = new User()
  15. .setUsername("xudaxian")
  16. .setEmail("xudaxian@qq.com")
  17. .setPassword("123456")
  18. .setName("许大仙")
  19. .setAge(18).insert();
  20. System.out.println("flag = " + flag);
  21. }
  22. }

1.5 更新数据

  • 示例:
  1. package com.example.mp;
  2. import com.example.mp.domain.User;
  3. import org.junit.jupiter.api.Test;
  4. import org.springframework.boot.test.context.SpringBootTest;
  5. /**
  6. * @author 许大仙
  7. * @version 1.0
  8. * @since 2021-06-19 18:47
  9. */
  10. @SpringBootTest
  11. public class TestActiveRecord {
  12. @Test
  13. public void testUpdate(){
  14. User user = new User();
  15. user.setId(1L);
  16. user.setAge(18);
  17. boolean flag = user.updateById();
  18. System.out.println("flag = " + flag);
  19. }
  20. }

1.6 删除数据

  • 示例:
  1. package com.example.mp;
  2. import com.example.mp.domain.User;
  3. import org.junit.jupiter.api.Test;
  4. import org.springframework.boot.test.context.SpringBootTest;
  5. /**
  6. * @author 许大仙
  7. * @version 1.0
  8. * @since 2021-06-19 18:47
  9. */
  10. @SpringBootTest
  11. public class TestActiveRecord {
  12. @Test
  13. public void testDelete(){
  14. User user = new User();
  15. user.setId(1L);
  16. boolean flag = user.deleteById();
  17. System.out.println("flag = " + flag);
  18. }
  19. }

1.7 根据条件查询数据

  • 示例:
  1. package com.example.mp;
  2. import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
  3. import com.example.mp.domain.User;
  4. import org.junit.jupiter.api.Test;
  5. import org.springframework.boot.test.context.SpringBootTest;
  6. import java.util.List;
  7. /**
  8. * @author 许大仙
  9. * @version 1.0
  10. * @since 2021-06-19 18:47
  11. */
  12. @SpringBootTest
  13. public class TestActiveRecord {
  14. @Test
  15. public void testSelect(){
  16. QueryWrapper<User> wrapper = new QueryWrapper<>();
  17. wrapper.le("age","20");
  18. User user = new User();
  19. List<User> userList = user.selectList(wrapper);
  20. System.out.println("userList = " + userList);
  21. }
  22. }

第二章:Mybatis Plus的插件

2.1 mybatis的插件机制

  • mybatis允许我们在已映射语句执行过程中的某一点进行拦截调用。默认情况下,Mybatis允许使用插件来拦截的方法调用包括:
  1. Executor (update, query, flushStatements, commit, rollback, getTransaction, close, isClosed)
  1. ParameterHandler (getParameterObject, setParameters)
  1. ResultSetHandler (handleResultSets, handleOutputParameters)
  1. StatementHandler (prepare, parameterize, batch, update, query)
  • 我们可以看到可以拦截Executor接口的部分方法,比如update、query、commit、rollback等方法,还有其他接口的一些方法等。
  • 总体概况为:
    • 拦截执行器的方法。
    • 拦截参数的处理。
    • 拦截结果集的处理。
    • 拦截Sql语法构建的处理。
  • 示例:
  • ①自定义拦截器
  1. package com.example.mp.plugins;
  2. import org.apache.ibatis.executor.statement.StatementHandler;
  3. import org.apache.ibatis.plugin.*;
  4. import java.sql.Statement;
  5. import java.util.Properties;
  6. /**
  7. * //定义拦截那个对象的那个方法的那个参数
  8. */
  9. @Intercepts({@Signature(type = StatementHandler.class, method = "parameterize", args = Statement.class)})
  10. public class MyInterceptor implements Interceptor {
  11. /**
  12. * 拦截目标对象的目标方法的执行
  13. *
  14. * @param invocation
  15. * @return
  16. * @throws Throwable
  17. */
  18. @Override
  19. public Object intercept(Invocation invocation) throws Throwable {
  20. System.out.println("--------MyInterceptor.intercept------------"+invocation.getMethod());
  21. //执行目标方法
  22. Object proceed = invocation.proceed();
  23. return proceed;
  24. }
  25. /**
  26. * 包装目标对象:为目标对象创建一个代理对象
  27. *
  28. * @param target
  29. * @return
  30. */
  31. @Override
  32. public Object plugin(Object target) {
  33. System.out.println("------MyInterceptor--plugin-------"+target);
  34. //借助Plugin的wrap使用当前的拦截器包装目标对象
  35. Object wrap = Plugin.wrap(target, this);
  36. //为当前target创建的动态代理
  37. return wrap;
  38. }
  39. /**
  40. * 将插件注册时的properties属性设置进来
  41. *
  42. * @param properties
  43. */
  44. @Override
  45. public void setProperties(Properties properties) {
  46. System.out.println("插件配置的信息 = " + properties);
  47. }
  48. }
  • ②注册到Spring的容器中:
  1. package com.example.mp.config;
  2. import com.baomidou.mybatisplus.annotation.DbType;
  3. import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
  4. import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor;
  5. import com.example.mp.plugins.MyInterceptor;
  6. import org.springframework.context.annotation.Bean;
  7. import org.springframework.context.annotation.Configuration;
  8. /**
  9. * @author 许大仙
  10. * @version 1.0
  11. * @since 2021-06-18 22:42
  12. */
  13. @Configuration
  14. public class MybatisPlusConfig {
  15. // 配置分页插件
  16. @Bean
  17. public MybatisPlusInterceptor mybatisPlusInterceptor() {
  18. MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
  19. interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));
  20. return interceptor;
  21. }
  22. /**
  23. * 注册 插件
  24. *
  25. * @return
  26. */
  27. @Bean
  28. public MyInterceptor myInterceptor() {
  29. return new MyInterceptor();
  30. }
  31. }

2.2 防止全表更新与删除插件

  • 在MP中提供了防止全表更新与删除插件。

注意:此插件仅适用于开发环境,不适用于生产环境。

  • 示例:
  • ①配置防止全表更新与删除的插件
  1. package com.example.mp.config;
  2. import com.baomidou.mybatisplus.annotation.DbType;
  3. import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
  4. import com.baomidou.mybatisplus.extension.plugins.inner.BlockAttackInnerInterceptor;
  5. import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor;
  6. import org.springframework.context.annotation.Bean;
  7. import org.springframework.context.annotation.Configuration;
  8. /**
  9. * @author 许大仙
  10. * @version 1.0
  11. * @since 2021-06-18 22:42
  12. */
  13. @Configuration
  14. public class MybatisPlusConfig {
  15. @Bean
  16. public MybatisPlusInterceptor mybatisPlusInterceptor() {
  17. MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
  18. //配置分页插件
  19. PaginationInnerInterceptor paginationInnerInterceptor = new PaginationInnerInterceptor(DbType.MYSQL);
  20. interceptor.addInnerInterceptor(paginationInnerInterceptor);
  21. //配置防止全表更新与删除的插件
  22. BlockAttackInnerInterceptor blockAttackInnerInterceptor = new BlockAttackInnerInterceptor();
  23. interceptor.addInnerInterceptor(blockAttackInnerInterceptor);
  24. return interceptor;
  25. }
  26. }
  • ②测试:
  1. package com.example.mp;
  2. import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
  3. import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper;
  4. import com.example.mp.domain.User;
  5. import com.example.mp.mapper.UserMapper;
  6. import org.junit.jupiter.api.Test;
  7. import org.mybatis.spring.MyBatisSystemException;
  8. import org.springframework.beans.factory.annotation.Autowired;
  9. import org.springframework.boot.test.context.SpringBootTest;
  10. /**
  11. * @author 许大仙
  12. * @version 1.0
  13. * @since 2021-06-18 19:22
  14. */
  15. @SpringBootTest
  16. public class TestUpdate {
  17. @Autowired
  18. private UserMapper userMapper;
  19. @Test
  20. public void test() {
  21. //封装更新字段的对象
  22. User user = new User();
  23. user.setAge(18);
  24. try {
  25. int result = userMapper.update(user, null);
  26. System.out.println("result = " + result);
  27. } catch (MyBatisSystemException e) {
  28. e.printStackTrace();
  29. }
  30. }
  31. }
  • ③结果。

防止全表更新与删除插件.png

2.3 乐观锁插件

2.3.1 主要适用场景

  • 当要更新一条记录的时候,希望这条记录没有被别人更新。
  • 乐观锁实现方式:
    • ①取出记录时,获取当前version。
    • ②更新时,带上这个version。
    • ③执行更新时, set version = newVersion where version = oldVersion
    • ④如果version不对,就更新失败。

2.3.2 插件配置

  • 乐观锁配置:
  1. package com.example.mp.config;
  2. import com.baomidou.mybatisplus.annotation.DbType;
  3. import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
  4. import com.baomidou.mybatisplus.extension.plugins.inner.BlockAttackInnerInterceptor;
  5. import com.baomidou.mybatisplus.extension.plugins.inner.OptimisticLockerInnerInterceptor;
  6. import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor;
  7. import org.springframework.context.annotation.Bean;
  8. import org.springframework.context.annotation.Configuration;
  9. /**
  10. * @author 许大仙
  11. * @version 1.0
  12. * @since 2021-06-18 22:42
  13. */
  14. @Configuration
  15. public class MybatisPlusConfig {
  16. @Bean
  17. public MybatisPlusInterceptor mybatisPlusInterceptor() {
  18. MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
  19. //配置分页插件
  20. PaginationInnerInterceptor paginationInnerInterceptor = new PaginationInnerInterceptor(DbType.MYSQL);
  21. interceptor.addInnerInterceptor(paginationInnerInterceptor);
  22. //乐观锁配置
  23. OptimisticLockerInnerInterceptor optimisticLockerInnerInterceptor = new OptimisticLockerInnerInterceptor();
  24. interceptor.addInnerInterceptor(optimisticLockerInnerInterceptor);
  25. //配置防止全表更新与删除的插件
  26. BlockAttackInnerInterceptor blockAttackInnerInterceptor = new BlockAttackInnerInterceptor();
  27. interceptor.addInnerInterceptor(blockAttackInnerInterceptor);
  28. return interceptor;
  29. }
  30. }

2.3.3 注解实体字段

  • 需要为实体字段添加@Version注解。
  • ①修改表,添加version字段,并且设置初始值为1。
  1. ALTER TABLE `tb_user` ADD COLUMN `version` int NULL ;
  1. UPDATE `tb_user` SET `version`='1';
  • ②为User实体对象添加version字段,并且添加@Version注解。
  1. package com.example.mp.domain;
  2. import com.baomidou.mybatisplus.annotation.IdType;
  3. import com.baomidou.mybatisplus.annotation.TableField;
  4. import com.baomidou.mybatisplus.annotation.TableId;
  5. import com.baomidou.mybatisplus.annotation.Version;
  6. import lombok.AllArgsConstructor;
  7. import lombok.Data;
  8. import lombok.NoArgsConstructor;
  9. import lombok.ToString;
  10. import lombok.experimental.Accessors;
  11. import java.io.Serializable;
  12. /**
  13. * @author 许大仙
  14. * @version 1.0
  15. * @since 2021-06-17 21:01
  16. */
  17. @Data
  18. @NoArgsConstructor
  19. @AllArgsConstructor
  20. //@TableName("tb_user")
  21. @ToString
  22. @Accessors(chain = true)
  23. public class User implements Serializable {
  24. @TableId(type= IdType.AUTO)
  25. private Long id;
  26. @TableField("user_name") //解决字段名不一致
  27. private String username;
  28. private String password;
  29. private String name;
  30. private Integer age;
  31. private String email;
  32. @Version
  33. private Integer version;
  34. }

2.3.4 测试

  • 测试:
  1. package com.example.mp;
  2. import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
  3. import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper;
  4. import com.example.mp.domain.User;
  5. import com.example.mp.mapper.UserMapper;
  6. import org.junit.jupiter.api.Test;
  7. import org.mybatis.spring.MyBatisSystemException;
  8. import org.springframework.beans.factory.annotation.Autowired;
  9. import org.springframework.boot.test.context.SpringBootTest;
  10. /**
  11. * @author 许大仙
  12. * @version 1.0
  13. * @since 2021-06-18 19:22
  14. */
  15. @SpringBootTest
  16. public class TestUpdate {
  17. @Autowired
  18. private UserMapper userMapper;
  19. @Test
  20. public void testOptimisticLocker() {
  21. //先查询再修改,会触发乐观锁配置,即实体对象中要有version字段值
  22. User user = userMapper.selectById(1);
  23. user.setUsername("许大仙");
  24. int result = userMapper.updateById(user);
  25. System.out.println("result = " + result);
  26. }
  27. }

乐观锁插件.png

2.3.5 特别说明

  • 支持的数据类型只有:int、Integer、long、Long、Date、Timestamp、LocalDateTime。
  • 整数类型下 newVersion = oldVersion + 1
  • newVersion 会回写到 entity
  • 仅支持 updateById(id)update(entity, wrapper) 方法
  • update(entity, wrapper) 方法下, wrapper 不能复用!!!

第三章:SQL注入器

3.1 概述

  • 我们知道,在MP中,通过AbstractSqlInjector将BaseMapper中的方法注入到Mybatis容器中,这样这些方法才可以正常执行。
  • 那么,我们如果需要扩充BaseMapper中的方法,又该如何实现。

3.2 编写MyBaseMapper

  • MyBaseMapper.java
  1. package com.example.mp.config;
  2. import com.baomidou.mybatisplus.core.mapper.BaseMapper;
  3. import java.util.List;
  4. public interface MyBaseMapper<T> extends BaseMapper<T> {
  5. List<T> findALl();
  6. }
  • 其他的Mapper都可以继承该Mapper,这样实现了统一的扩展:
  1. package com.example.mp.mapper;
  2. import com.example.mp.config.MyBaseMapper;
  3. import com.example.mp.domain.User;
  4. import org.apache.ibatis.annotations.Mapper;
  5. /**
  6. * @author 许大仙
  7. * @version 1.0
  8. * @since 2021-06-17 21:10
  9. */
  10. @Mapper
  11. public interface UserMapper extends MyBaseMapper<User> {
  12. }

3.3 编写FindAll

  1. package com.example.mp.config;
  2. import com.baomidou.mybatisplus.core.injector.AbstractMethod;
  3. import com.baomidou.mybatisplus.core.metadata.TableInfo;
  4. import org.apache.ibatis.mapping.MappedStatement;
  5. import org.apache.ibatis.mapping.SqlSource;
  6. public class FindAll extends AbstractMethod {
  7. @Override
  8. public MappedStatement injectMappedStatement(Class<?> mapperClass, Class<?> modelClass, TableInfo tableInfo) {
  9. String sqlMethod = "findAll";
  10. String sql = "select * from " + tableInfo.getTableName();
  11. SqlSource sqlSource = languageDriver.createSqlSource(configuration, sql,
  12. modelClass);
  13. return this.addSelectMappedStatementForTable(mapperClass, sqlMethod, sqlSource, tableInfo);
  14. }
  15. }

3.4 编写MySqlInjector

  • 如果直接继承AbstractSqlInjector的话,原有的BaseMapper中的方法将失效,我们选择继承DefaultSqlInjector来实现扩展。
  1. package com.example.mp.config;
  2. import com.baomidou.mybatisplus.core.injector.AbstractMethod;
  3. import com.baomidou.mybatisplus.core.injector.DefaultSqlInjector;
  4. import org.springframework.stereotype.Component;
  5. import java.util.List;
  6. @Component
  7. public class MySqlInjector extends DefaultSqlInjector {
  8. @Override
  9. public List<AbstractMethod> getMethodList(Class<?> mapperClass) {
  10. List<AbstractMethod> methodList = super.getMethodList(mapperClass);
  11. methodList.add(new FindAll());
  12. return methodList;
  13. }
  14. }

3.5 测试

  1. package com.example.mp;
  2. import com.example.mp.domain.User;
  3. import com.example.mp.mapper.UserMapper;
  4. import org.junit.jupiter.api.Test;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.boot.test.context.SpringBootTest;
  7. import java.util.List;
  8. @SpringBootTest
  9. class MpApplicationTests {
  10. @Autowired
  11. private UserMapper userMapper;
  12. @Test
  13. public void test(){
  14. List<User> userList = userMapper.findAll();
  15. System.out.println("userList = " + userList);
  16. }
  17. }

第四章:自动填充

4.1 概述

  • 有些时候,我们可能会有这样的需求,插入或更新数据的时候,希望有些字段可以自动填充数据,比如:密码、version等。在MP中提供了这样的功能,可能实现自动填充。

4.2 添加@TableField注解

  • User.java
  1. package com.example.mp.domain;
  2. import com.baomidou.mybatisplus.annotation.*;
  3. import lombok.AllArgsConstructor;
  4. import lombok.Data;
  5. import lombok.NoArgsConstructor;
  6. import lombok.ToString;
  7. import lombok.experimental.Accessors;
  8. import java.io.Serializable;
  9. import java.util.Date;
  10. /**
  11. * @author 许大仙
  12. * @version 1.0
  13. * @since 2021-06-17 21:01
  14. */
  15. @Data
  16. @NoArgsConstructor
  17. @AllArgsConstructor
  18. @ToString
  19. @Accessors(chain = true)
  20. public class User implements Serializable {
  21. @TableId(type = IdType.AUTO)
  22. private Long id;
  23. @TableField("user_name") //解决字段名不一致
  24. private String username;
  25. private String password;
  26. private String name;
  27. private Integer age;
  28. private String email;
  29. @Version
  30. private Integer version;
  31. /**
  32. * 为createTime添加自动填充功能,在新增数据的时候有效
  33. */
  34. @TableField(value = "create_time",fill = FieldFill.INSERT)
  35. private Date createTime;
  36. /**
  37. * 为updateTime添加自动填充功能,在更新数据的时候有效
  38. */
  39. @TableField(value = "update_time",fill = FieldFill.UPDATE)
  40. private Date updateTime;
  41. }
  • sql脚本:
  1. ALTER TABLE `tb_user` ADD COLUMN `create_time` datetime NULL ;
  1. ALTER TABLE `tb_user` ADD COLUMN `update_time` datetime NULL ;
  • FieldFill提供了多种模式选择:
  1. public enum FieldFill {
  2. /**
  3. * 默认不处理
  4. */
  5. DEFAULT,
  6. /**
  7. * 插入时填充字段
  8. */
  9. INSERT,
  10. /**
  11. * 更新时填充字段
  12. */
  13. UPDATE,
  14. /**
  15. * 插入和更新时填充字段
  16. */
  17. INSERT_UPDATE
  18. }

4.3 自定义MetaObjectHandler

  1. package com.example.mp.config;
  2. import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler;
  3. import lombok.extern.slf4j.Slf4j;
  4. import org.apache.ibatis.reflection.MetaObject;
  5. import org.springframework.stereotype.Component;
  6. import java.util.Date;
  7. @Slf4j
  8. @Component
  9. public class MPMetaObjectHandler implements MetaObjectHandler {
  10. @Override
  11. public void insertFill(MetaObject metaObject) {
  12. this.strictInsertFill(metaObject, "createTime", () -> new Date(), Date.class);
  13. }
  14. @Override
  15. public void updateFill(MetaObject metaObject) {
  16. this.strictUpdateFill(metaObject, "updateTime", () -> new Date(), Date.class);
  17. }
  18. }

4.4 测试

  • 示例:
  1. package com.example.mp;
  2. import com.example.mp.domain.User;
  3. import com.example.mp.mapper.UserMapper;
  4. import org.junit.jupiter.api.Test;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.boot.test.context.SpringBootTest;
  7. import java.util.List;
  8. @SpringBootTest
  9. class MpApplicationTests {
  10. @Autowired
  11. private UserMapper userMapper;
  12. @Test
  13. public void testInsert(){
  14. User user = new User();
  15. user.setUsername("zhangsan");
  16. user.setPassword("123456");
  17. user.setName("张三");
  18. user.setAge(18);
  19. user.setEmail("123456@qq.com");
  20. int count = userMapper.insert(user);
  21. System.out.println("count = " + count); //数据库受影响的行数
  22. System.out.println("user.getId() = " + user.getId()); //自增后的id会回填到对象中
  23. }
  24. }
  • 结果:

自动填充结果.png

第五章:逻辑删除

5.1 概述

  • 开发系统的时候,有时候在实现功能的时候,删除操作需要实现逻辑删除,所谓的逻辑删除就是将数据标记为删除,而非真正的物理删除(非DELETE操作),查询的时候需要携带状态条件,确保被标记的数据不被查询到。这样做的目的就是避免数据被真正的删除。

5.2 修改表结构

  • tb_user表添加deleted字段,用于表示数据是否被删除,1代表删除,0代表未删除。
  1. ALTER TABLE `tb_user` ADD COLUMN `deleted` INT ( 1 ) NULL DEFAULT 0 COMMENT '1代表删除,0代表未删除' AFTER `version`;
  • 修改User实体,增加deleted属性,并添加@TableLogic注解。
  1. package com.example.mp.domain;
  2. import com.baomidou.mybatisplus.annotation.*;
  3. import lombok.AllArgsConstructor;
  4. import lombok.Data;
  5. import lombok.NoArgsConstructor;
  6. import lombok.ToString;
  7. import lombok.experimental.Accessors;
  8. import java.io.Serializable;
  9. import java.util.Date;
  10. /**
  11. * @author 许大仙
  12. * @version 1.0
  13. * @since 2021-06-17 21:01
  14. */
  15. @Data
  16. @NoArgsConstructor
  17. @AllArgsConstructor
  18. @ToString
  19. @Accessors(chain = true)
  20. public class User implements Serializable {
  21. @TableId(type = IdType.AUTO)
  22. private Long id;
  23. @TableField("user_name") //解决字段名不一致
  24. private String username;
  25. private String password;
  26. private String name;
  27. private Integer age;
  28. private String email;
  29. @Version
  30. private Integer version;
  31. /**
  32. * 为createTime添加自动填充功能,在新增数据的时候有效
  33. */
  34. @TableField(value = "create_time",fill = FieldFill.INSERT)
  35. private Date createTime;
  36. /**
  37. * 为updateTime添加自动填充功能,在更新数据的时候有效
  38. */
  39. @TableField(value = "update_time",fill = FieldFill.UPDATE)
  40. private Date updateTime;
  41. @TableLogic
  42. private Integer deleted;
  43. }

5.3 配置

  • application.yml
  1. # MP配置
  2. mybatis-plus:
  3. global-config:
  4. db-config:
  5. table-prefix: tb_
  6. logic-delete-field: deleted #全局逻辑删除的实体字段名
  7. logic-delete-value: 1 # 逻辑已删除值
  8. logic-not-delete-value: 0 # 逻辑未删除值
  9. configuration:
  10. # 开启日志
  11. log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

5.4 测试

  • 测试删除:
  1. package com.example.mp;
  2. import com.example.mp.mapper.UserMapper;
  3. import org.junit.jupiter.api.Test;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.boot.test.context.SpringBootTest;
  6. /**
  7. * @author 许大仙
  8. * @version 1.0
  9. * @since 2021-06-18 20:04
  10. */
  11. @SpringBootTest
  12. public class TestDeleteById {
  13. @Autowired
  14. private UserMapper userMapper;
  15. @Test
  16. public void test(){
  17. //执行删除操作
  18. int result = userMapper.deleteById(6);
  19. System.out.println("result = " + result);
  20. }
  21. }
  • 结果:

测试逻辑删除的结果.png

  • 测试查询:
  1. package com.example.mp;
  2. import com.example.mp.domain.User;
  3. import com.example.mp.mapper.UserMapper;
  4. import org.junit.jupiter.api.Test;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.boot.test.context.SpringBootTest;
  7. /**
  8. * @author 许大仙
  9. * @version 1.0
  10. * @since 2021-06-18 21:00
  11. */
  12. @SpringBootTest
  13. public class TestSelectById {
  14. @Autowired
  15. private UserMapper userMapper;
  16. @Test
  17. public void test(){
  18. //根据id查询数据
  19. User user = userMapper.selectById(6);
  20. System.out.println("user = " + user);
  21. }
  22. }
  • 结果:

测试逻辑删除查询的结果.png

第六章:通用枚举

6.1 概述

  • 解决了繁琐的配置,让 mybatis 优雅的使用枚举属性!

6.2 修改表结构

  1. ALTER TABLE `tb_user`
  2. ADD COLUMN `sex` int(1) NULL DEFAULT 1 COMMENT '1-男,2-女' AFTER `deleted`;

6.3 定义枚举

  1. package com.example.mp.enums;
  2. import com.baomidou.mybatisplus.annotation.IEnum;
  3. public enum SexEnum implements IEnum<Integer> {
  4. MAN(1,"男"),
  5. WOMAN(2,"女");
  6. private int value;
  7. private String desc;
  8. SexEnum(int value, String desc) {
  9. this.value = value;
  10. this.desc = desc;
  11. }
  12. @Override
  13. public Integer getValue() {
  14. return value;
  15. }
  16. }

6.4 配置

  • application.yml
  1. # MP配置
  2. mybatis-plus:
  3. global-config:
  4. db-config:
  5. table-prefix: tb_
  6. logic-delete-field: deleted #全局逻辑删除的实体字段名
  7. logic-delete-value: 1 # 逻辑已删除值
  8. logic-not-delete-value: 0 # 逻辑未删除值
  9. configuration:
  10. # 开启日志
  11. log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
  12. # 配置枚举包扫描
  13. type-enums-package: com.example.mp.enums

6.5 修改实体

  1. package com.example.mp.domain;
  2. import com.baomidou.mybatisplus.annotation.*;
  3. import com.example.mp.enums.SexEnum;
  4. import lombok.AllArgsConstructor;
  5. import lombok.Data;
  6. import lombok.NoArgsConstructor;
  7. import lombok.ToString;
  8. import lombok.experimental.Accessors;
  9. import java.io.Serializable;
  10. import java.util.Date;
  11. /**
  12. * @author 许大仙
  13. * @version 1.0
  14. * @since 2021-06-17 21:01
  15. */
  16. @Data
  17. @NoArgsConstructor
  18. @AllArgsConstructor
  19. @ToString
  20. @Accessors(chain = true)
  21. public class User implements Serializable {
  22. @TableId(type = IdType.AUTO)
  23. private Long id;
  24. @TableField("user_name") //解决字段名不一致
  25. private String username;
  26. private String password;
  27. private String name;
  28. private Integer age;
  29. private String email;
  30. //性别为枚举
  31. private SexEnum sex;
  32. @Version
  33. @TableField(value = "version", fill = FieldFill.INSERT)
  34. private Integer version;
  35. /**
  36. * 为createTime添加自动填充功能,在新增数据的时候有效
  37. */
  38. @TableField(value = "create_time", fill = FieldFill.INSERT)
  39. private Date createTime;
  40. /**
  41. * 为updateTime添加自动填充功能,在更新数据的时候有效
  42. */
  43. @TableField(value = "update_time", fill = FieldFill.UPDATE)
  44. private Date updateTime;
  45. @TableLogic
  46. private Integer deleted;
  47. }

6.6 测试

  • 示例:
  1. package com.example.mp;
  2. import com.example.mp.domain.User;
  3. import com.example.mp.enums.SexEnum;
  4. import com.example.mp.mapper.UserMapper;
  5. import org.junit.jupiter.api.Test;
  6. import org.springframework.beans.factory.annotation.Autowired;
  7. import org.springframework.boot.test.context.SpringBootTest;
  8. import java.util.List;
  9. @SpringBootTest
  10. class MpApplicationTests {
  11. @Autowired
  12. private UserMapper userMapper;
  13. @Test
  14. public void testInsert(){
  15. User user = new User();
  16. user.setUsername("wangba");
  17. user.setPassword("123456");
  18. user.setName("王八");
  19. user.setAge(18);
  20. user.setEmail("123456@qq.com");
  21. user.setSex(SexEnum.MAN);
  22. int count = userMapper.insert(user);
  23. System.out.println("count = " + count); //数据库受影响的行数
  24. System.out.println("user.getId() = " + user.getId()); //自增后的id会回填到对象中
  25. }
  26. }
  • 结果:

通用枚举结果.png

第七章:代码生成器

7.1 概述

  • AutoGenerator 是 MyBatis-Plus 的代码生成器,通过 AutoGenerator 可以快速生成 Entity、Mapper、Mapper XML、Service、Controller 等各个模块的代码,极大的提升了开发效率。

7.2 创建工程

  • pom.xml
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
  4. <modelVersion>4.0.0</modelVersion>
  5. <parent>
  6. <groupId>org.springframework.boot</groupId>
  7. <artifactId>spring-boot-starter-parent</artifactId>
  8. <version>2.5.1</version>
  9. <relativePath/>
  10. </parent>
  11. <groupId>com.example</groupId>
  12. <artifactId>mp-generator</artifactId>
  13. <version>0.0.1-SNAPSHOT</version>
  14. <name>mp-generator</name>
  15. <description>Demo project for Spring Boot</description>
  16. <properties>
  17. <java.version>11</java.version>
  18. </properties>
  19. <dependencies>
  20. <dependency>
  21. <groupId>org.springframework.boot</groupId>
  22. <artifactId>spring-boot-starter-web</artifactId>
  23. </dependency>
  24. <dependency>
  25. <groupId>org.projectlombok</groupId>
  26. <artifactId>lombok</artifactId>
  27. <optional>true</optional>
  28. </dependency>
  29. <dependency>
  30. <groupId>mysql</groupId>
  31. <artifactId>mysql-connector-java</artifactId>
  32. <scope>runtime</scope>
  33. </dependency>
  34. <dependency>
  35. <groupId>org.springframework.boot</groupId>
  36. <artifactId>spring-boot-starter-test</artifactId>
  37. <scope>test</scope>
  38. </dependency>
  39. <!-- mybatis-plus对SpringBoot的支持 -->
  40. <dependency>
  41. <groupId>com.baomidou</groupId>
  42. <artifactId>mybatis-plus-boot-starter</artifactId>
  43. <version>3.4.3.1</version>
  44. </dependency>
  45. <dependency>
  46. <groupId>com.baomidou</groupId>
  47. <artifactId>mybatis-plus-generator</artifactId>
  48. <version>3.4.1</version>
  49. </dependency>
  50. <dependency>
  51. <groupId>org.freemarker</groupId>
  52. <artifactId>freemarker</artifactId>
  53. <version>2.3.31</version>
  54. </dependency>
  55. </dependencies>
  56. <build>
  57. <plugins>
  58. <plugin>
  59. <groupId>org.springframework.boot</groupId>
  60. <artifactId>spring-boot-maven-plugin</artifactId>
  61. </plugin>
  62. </plugins>
  63. </build>
  64. </project>

7.3 生成器代码

  1. package com.example.mp.generator;
  2. import com.baomidou.mybatisplus.core.exceptions.MybatisPlusException;
  3. import com.baomidou.mybatisplus.core.toolkit.StringUtils;
  4. import com.baomidou.mybatisplus.generator.AutoGenerator;
  5. import com.baomidou.mybatisplus.generator.config.DataSourceConfig;
  6. import com.baomidou.mybatisplus.generator.config.GlobalConfig;
  7. import com.baomidou.mybatisplus.generator.config.PackageConfig;
  8. import com.baomidou.mybatisplus.generator.config.StrategyConfig;
  9. import com.baomidou.mybatisplus.generator.config.rules.NamingStrategy;
  10. import com.baomidou.mybatisplus.generator.engine.FreemarkerTemplateEngine;
  11. import java.util.Scanner;
  12. public class MysqlGenerator {
  13. /**
  14. * <p>
  15. * 读取控制台内容
  16. * </p>
  17. */
  18. public static String scanner(String tip) {
  19. Scanner scanner = new Scanner(System.in);
  20. StringBuilder help = new StringBuilder();
  21. help.append("请输入" + tip + ":");
  22. System.out.println(help);
  23. if (scanner.hasNext()) {
  24. String ipt = scanner.next();
  25. if (StringUtils.isNotBlank(ipt)) {
  26. return ipt;
  27. }
  28. }
  29. throw new MybatisPlusException("请输入正确的" + tip + "!");
  30. }
  31. /**
  32. * RUN THIS
  33. */
  34. public static void main(String[] args) {
  35. // 代码生成器
  36. AutoGenerator mpg = new AutoGenerator();
  37. // 全局配置
  38. GlobalConfig gc = new GlobalConfig();
  39. String projectPath = System.getProperty("user.dir");
  40. gc.setOutputDir(projectPath + "/src/main/java");
  41. gc.setAuthor("许大仙");
  42. gc.setOpen(false);
  43. // gc.setSwagger2(true); 实体属性 Swagger2 注解
  44. mpg.setGlobalConfig(gc);
  45. // 数据源配置
  46. DataSourceConfig dsc = new DataSourceConfig();
  47. dsc.setUrl("jdbc:mysql://127.0.0.1:3306/mp?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&useSSL=false&serverTimezone=GMT%2B8&allowPublicKeyRetrieval=true&nullCatalogMeansCurrent=true");
  48. // dsc.setSchemaName("public");
  49. dsc.setDriverName("com.mysql.cj.jdbc.Driver");
  50. dsc.setUsername("root");
  51. dsc.setPassword("123456");
  52. mpg.setDataSource(dsc);
  53. // 包配置
  54. PackageConfig pc = new PackageConfig();
  55. pc.setModuleName(scanner("模块名"));
  56. pc.setParent("com.example.mp.generator");
  57. mpg.setPackageInfo(pc);
  58. // // 自定义配置
  59. // InjectionConfig cfg = new InjectionConfig() {
  60. // @Override
  61. // public void initMap() {
  62. // // to do nothing
  63. // }
  64. // };
  65. // List<FileOutConfig> focList = new ArrayList<>();
  66. // focList.add(new FileOutConfig("/templates/mapper.xml.ftl") {
  67. // @Override
  68. // public String outputFile(TableInfo tableInfo) {
  69. // // 自定义输入文件名称
  70. // return projectPath + "/xudaxain-mp-generator/src/main/resources/mapper/" + pc.getModuleName()
  71. // + "/" + tableInfo.getEntityName() + "Mapper" + StringPool.DOT_XML;
  72. // }
  73. // });
  74. // cfg.setFileOutConfigList(focList);
  75. // mpg.setCfg(cfg);
  76. // mpg.setTemplate(new TemplateConfig().setXml(null));
  77. // 策略配置
  78. StrategyConfig strategy = new StrategyConfig();
  79. strategy.setNaming(NamingStrategy.underline_to_camel);
  80. strategy.setColumnNaming(NamingStrategy.underline_to_camel);
  81. // strategy.setSuperEntityClass("com.baomidou.mybatisplus.samples.generator.common.BaseEntity");
  82. strategy.setEntityLombokModel(true);
  83. // strategy.setSuperControllerClass("com.baomidou.mybatisplus.samples.generator.common.BaseController");
  84. strategy.setInclude(scanner("表名"));
  85. strategy.setSuperEntityColumns("id");
  86. strategy.setControllerMappingHyphenStyle(true);
  87. strategy.setTablePrefix(pc.getModuleName() + "_");
  88. mpg.setStrategy(strategy);
  89. // 选择 freemarker 引擎需要指定如下加,注意 pom 依赖必须有!
  90. mpg.setTemplateEngine(new FreemarkerTemplateEngine());
  91. mpg.execute();
  92. }
  93. }