Springboot入门之集成MybatisPlus

MybatisPlus是一个 Mybatis的增强工具,在Mybatis的基础上只做增强不做改变,为简化开发、提高效率而生。 

1.添加依赖【pom.xml】

        
        
            mysql
            mysql-connector-java
        

        
            com.baomidou
            mybatis-plus-boot-starter
            3.5.1
        

        
            com.baomidou
            mybatis-plus-annotation
            3.5.1
            compile
        

        
        
            com.baomidou
            mybatis-plus-generator
            3.5.1
        

2.配置数据源信息[application.yml]

# 数据源配置
spring:
	datasource:
		url: jdbc:mysql://xxx:3306/xxx?useUnicode=true&characterEncoding=UTF-8&serverTimezone=Asia/Shanghai&useSSL=false&allowMultiQueries=true
		username: xxx
		password: xxxxx
		dbcp2:
			validation-query: SELECT 1
			time-between-eviction-runs-millis: 18800
		hikari:
			max-lifetime: 30000
			validation-timeout: 3000
			connection-timeout: 18800
			idle-timeout: 18800

# sql日志输出配置
mybatis-plus:
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

3.MybatisPlus拦截器配置

配置后MybatisPlus增加功能生效【如分页等】。

@Configuration
@MapperScan(value = {"com.{company}.{model}.mapper"})
public class MybatisPlusConfig {

    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
        interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));
        return interceptor;
    }

}

4.编写XxxMapper

extends BaseMapper

/**
 * 

* Mapper 接口 *

* * @author ocean * @since 2023-03-17 */ @Mapper public interface XxxMapper extends BaseMapper { }

5.编写XxxService

extends ServiceImpl

@Service
public class XxxServiceImpl extends ServiceImpl implements XxxService {

    @Resource
    private XxxMapper XxxMapper;

    public void test(){
        XxxMapper.insert(new XxxDo());
        XxxMapper.deleteById(1L);
        XxxMapper.updateById(new XxxDo());
        XxxMapper.selectPage(new Page(1, 10), new QueryWrapper<>());
    }
}

6.Wapper查询

1.查询Wrapper
QueryWrapper queryWrapper = Wrappers.query();
LambdaQueryWrapper lambdaQueryWrapper = Wrappers.lambdaQuery();

2.更新Wrapper
UpdateWrapper updateWrapper = Wrappers.update();
LambdaUpdateWrapper lambdaUpdateWrapper = Wrappers.lambdaUpdate();

7.配置安全

7.1)生成数据源加密信息

import com.baomidou.mybatisplus.core.toolkit.AES;

public class JdbcEncodeUtil {

    public static final String URL = "jdbc:mysql://xxx:3306/xxx?useUnicode=true&characterEncoding=UTF-8&serverTimezone=Asia/Shanghai&useSSL=false&allowMultiQueries=true";
    public static final String USERNAME = "xxx";
    public static final String PASSWORD = "xxxx";

    public static void main(String[] args) {
        // 生成 16 位随机 AES 密钥
        String randomKey = AES.generateRandomKey();

        // 随机密钥加密
        String url = AES.encrypt(URL, randomKey);
        String username = AES.encrypt(USERNAME, randomKey);
        String password = AES.encrypt(PASSWORD, randomKey);

        System.out.println("randomKey = " + randomKey);
        System.out.println("url = " + url);
        System.out.println("username = " + username);
        System.out.println("password = " + password);

    }

}

7.2)配置数据源加密【application.yml】

// 加密配置 mpw: 开头紧接加密内容( 非数据库配置专用 YML 中其它配置也是可以使用的 )
spring:
  datasource:
    url: mpw:qRhvCwF4GOqjessEB3G+a5okP+uXXr96wcucn2Pev6Bf1oEMZ1gVpPPhdDmjQqoM
    password: mpw:Hzy5iliJbwDHhjLs1L0j6w==
    username: mpw:Xb+EgsyuYRXw7U7sBJjBpA==

7.3)配置秘钥

7.3.1)jar包运行 java -jar xxx.jar --mpw.key=1111111111111 

7.3.2)IDEA 设置 Program arguments,如下图:

Springboot入门之集成MybatisPlus_第1张图片

7.4)实现原理

SpringBoot 提供修改Spring环境后置处理器【EnvironmentPostProcessor】,允许在应用程序之前操作环境属性值。

public class SafetyEncryptProcessor implements EnvironmentPostProcessor {
    public SafetyEncryptProcessor() {
    }

    public void postProcessEnvironment(ConfigurableEnvironment environment, SpringApplication application) {
        String mpwKey = null;
        Iterator var4 = environment.getPropertySources().iterator();

        while(var4.hasNext()) {
            PropertySource ps = (PropertySource)var4.next();
            if (ps instanceof SimpleCommandLinePropertySource) {
                SimpleCommandLinePropertySource source = (SimpleCommandLinePropertySource)ps;
                mpwKey = source.getProperty("mpw.key");
                break;
            }
        }

        if (StringUtils.isNotBlank(mpwKey)) {
            HashMap map = new HashMap();
            Iterator var15 = environment.getPropertySources().iterator();

            while(true) {
                PropertySource ps;
                do {
                    if (!var15.hasNext()) {
                        if (CollectionUtils.isNotEmpty(map)) {
                            environment.getPropertySources().addFirst(new MapPropertySource("custom-encrypt", map));
                        }

                        return;
                    }

                    ps = (PropertySource)var15.next();
                } while(!(ps instanceof OriginTrackedMapPropertySource));

                OriginTrackedMapPropertySource source = (OriginTrackedMapPropertySource)ps;
                String[] var8 = source.getPropertyNames();
                int var9 = var8.length;

                for(int var10 = 0; var10 < var9; ++var10) {
                    String name = var8[var10];
                    Object value = source.getProperty(name);
                    if (value instanceof String) {
                        String str = (String)value;
                        if (str.startsWith("mpw:")) {
                            map.put(name, AES.decrypt(str.substring(4), mpwKey));
                        }
                    }
                }
            }
        }
    }
}

8.自动填充功能

8.1)实现元对象处理器接口【com.baomidou.mybatisplus.core.handlers.MetaObjectHandler】

@Slf4j
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {

    @Override
    public void insertFill(MetaObject metaObject) {
        log.info("start insert fill ....");
        // 起始版本 3.3.3(推荐)
        this.strictInsertFill(metaObject, "createTime", () -> new Date(), Date.class);
        // 起始版本 3.3.0(推荐使用)
        // this.strictInsertFill(metaObject, "createTime", LocalDateTime.class, LocalDateTime.now());
        //
        // this.fillStrategy(metaObject, "createTime", LocalDateTime.now());
    }

    @Override
    public void updateFill(MetaObject metaObject) {
        log.info("start update fill ....");
        // 起始版本 3.3.3(推荐)
        this.strictUpdateFill(metaObject, "updateTime", () -> new Date(), Date.class);
        // 起始版本 3.3.0(推荐)
        // this.strictUpdateFill(metaObject, "updateTime", LocalDateTime.class, LocalDateTime.now());
        // 也可以使用(3.3.0 该方法有bug)
        // this.fillStrategy(metaObject, "updateTime", LocalDateTime.now()); // 也可以使用(3.3.0 该方法有bug)
    }

}

8.2)注解填充字段【@TableField(fill = FieldFill.INSERT/.INSERT_UPDATE/.UPDATE) 】

/**
 * 基础 [entity]
 *
 * @author ocean.liu
 */
@Data
public class BaseEntity {

    /**
     * 创建时间
     */
    @TableField(value = "create_time", fill = FieldFill.INSERT)
    protected Date createTime;

    /**
     * 修改时间
     */
    @TableField(value = "update_time", fill = FieldFill.INSERT_UPDATE)
    protected Date updateTime;

    /**
     * 是否删除
     */
    @TableField(value = "is_deleted")
    protected Integer isDeleted;

}

9.代码生成器

public class CodeGenerator {


    /**
     * 数据库链接地址
     **/
    private static final String JDBC_URL_MAN = "jdbc:mysql://xxxxx:3306/xxx?useUnicode=true&characterEncoding=UTF-8";
    /**
     * 数据库登录账号
     **/
    private static final String JDBC_USER_NAME = "xx";
    /**
     * 数据库登录密码
     **/
    private static final String JDBC_PASSWORD = "xxxx";


    public static void main(String[] args) {
        String dir = "\\xx\\xxx";
        String tablePrefix = "t_";
        List tables = new ArrayList<>();
        tables.add("t_doctor");

        FastAutoGenerator.create(JDBC_URL_MAN, JDBC_USER_NAME, JDBC_PASSWORD)
                .globalConfig(builder -> {
                    builder.author("ocean")                                                           // 作者
                            .outputDir(System.getProperty("user.dir") + dir + "\\src\\main\\java")    // 输出路径(写到java目录)
                            .enableSwagger()                                                          // 开启swagger
                            .commentDate("yyyy-MM-dd")
                            .fileOverride();                                                          // 开启覆盖之前生成的文件

                })
                .packageConfig(builder -> {
                    builder.parent("com.{company}")
                            .moduleName("{model}")
                            .entity("entity")
                            .service("service")
                            .serviceImpl("serviceImpl")
                            .controller("api")
                            .mapper("mapper")
                            .xml("mapper")
                            .pathInfo(Collections.singletonMap(OutputFile.mapperXml, System.getProperty("user.dir") + dir + "\\src\\main\\resources\\mapper"));
                })
                .strategyConfig(builder -> {
                    builder.addInclude(tables)
                            .addTablePrefix(tablePrefix)
                            .serviceBuilder()
                            .formatServiceFileName("%sService")
                            .formatServiceImplFileName("%sServiceImpl")
                            .entityBuilder()
                            .enableLombok()
                            .logicDeleteColumnName("deleted")
                            .enableTableFieldAnnotation()
                            .controllerBuilder()
                            .formatFileName("%sController")
                            .enableRestStyle()
                            .mapperBuilder()
                            .enableBaseResultMap()               // 生成通用的resultMap
                            .superClass(BaseMapper.class)
                            .formatMapperFileName("%sMapper")
                            .enableMapperAnnotation()
                            .formatXmlFileName("%sMapper");
                })
                //.templateEngine(new FreemarkerTemplateEngine()) // 使用Freemarker引擎模板,默认的是Velocity引擎模板
                .execute();
    }

}

MyBatisPlus官方指南

你可能感兴趣的:(SpringBoot,spring,boot)