MyBatisPlus快速入门
需要的基础:
1.MyBatisPlus概述 MyBatis-Plus (简称 MP)是一个 MyBatis 的增强工具,在 MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。
愿景
是成为 MyBatis 最好的搭档,就像 魂斗罗 中的 1P、2P,基友搭配,效率翻倍。 特性
无侵入 :只做增强不做改变,引入它不会对现有工程产生影响,如丝般顺滑损耗小 :启动即会自动注入基本 CURD,性能基本无损耗,直接面向对象操作强大的 CRUD 操作 :内置通用 Mapper、通用 Service,仅仅通过少量配置即可实现单表大部分 CRUD 操作,更有强大的条件构造器,满足各类使用需求支持 Lambda 形式调用 :通过 Lambda 表达式,方便的编写各类查询条件,无需再担心字段写错支持主键自动生成 :支持多达 4 种主键策略(内含分布式唯一 ID 生成器 - Sequence),可自由配置,完美解决主键问题支持 ActiveRecord 模式 :支持 ActiveRecord 形式调用,实体类只需继承 Model 类即可进行强大的 CRUD 操作支持自定义全局通用操作 :支持全局通用方法注入( Write once, use anywhere )内置代码生成器 :采用代码或者 Maven 插件可快速生成 Mapper 、 Model 、 Service 、 Controller 层代码,支持模板引擎,更有超多自定义配置等您来使用内置分页插件 :基于 MyBatis 物理分页,开发者无需关心具体操作,配置好插件之后,写分页等同于普通 List 查询分页插件支持多种数据库 :支持 MySQL、MariaDB、Oracle、DB2、H2、HSQL、SQLite、Postgre、SQLServer 等多种数据库内置性能分析插件 :可输出 SQL 语句以及其执行时间,建议开发测试时启用该功能,能快速揪出慢查询内置全局拦截插件 :提供全表 delete 、 update 操作智能分析阻断,也可自定义拦截规则,预防误操作框架结构
代码托管
相关链接
教程
原理
2.快速入门 创建一个数据库mybatis_plus
操作如下语句,创建数据表,插入数据。 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 DROP TABLE IF EXISTS USER; 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) ); DELETE FROM 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'); -- 真实开发中,version(乐观锁)、deleted(逻辑删除)、gmt_create、gmt_modified
如果从零开始用 MyBatis-Plus 来实现该表的增删改查我们需要做什么呢?
导入对应的依赖
研究依赖如何配置
代码如何编写
提高扩展技术能力!
创建一个空的 Spring Boot 工程(工程将以 H2 作为默认数据库进行演示)
导入依赖,引入 spring-boot-starter
、spring-boot-starter-test
、mybatis-plus-boot-starter
、h2
依赖: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 <dependencies > <dependency > <groupId > mysql</groupId > <artifactId > mysql-connector-java</artifactId > </dependency > <dependency > <groupId > org.projectlombok</groupId > <artifactId > lombok</artifactId > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-web</artifactId > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter</artifactId > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-test</artifactId > <scope > test</scope > </dependency > <dependency > <groupId > com.baomidou</groupId > <artifactId > mybatis-plus-boot-starter</artifactId > <version > 3.0.5</version > </dependency > <dependency > <groupId > com.h2database</groupId > <artifactId > h2</artifactId > <scope > runtime</scope > </dependency > </dependencies >
说明:我们使用 mybatis-plus 可以节省我们大量的代码,尽量不要同时导入 mybatis 和 mybatis plus!版本的差异! 连接数据库!这一步和 mybatis 相同! 1 2 3 4 5 6 7 8 serverTimezone=GMT%2B8 spring.datasource.username=root spring.datasource.password=root spring.datasource.url=jdbc:mysql://localhost:3306/mybatis_plus?useSSL=false&useUnicode=true&characterEncoding=utf-8&serverTimezone=GMT%2B8 spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
==传统方式pojo-dao(连接mybatis,配置mapper.xml文件)-service-controller==,使用了mybatis-plus 之后: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 package com.github.pojo;import lombok.AllArgsConstructor;import lombok.Data;import lombok.NoArgsConstructor;@Data @AllArgsConstructor @NoArgsConstructor public class User { private Long id; private String name; private String email; private Integer age; }
1 2 3 4 5 6 7 8 9 10 package com.github.mapper;import com.baomidou.mybatisplus.core.mapper.BaseMapper;import com.github.pojo.User;import org.springframework.stereotype.Repository;@Repository public interface UserMapper extends BaseMapper <User> { }
在 Spring Boot 启动类中添加 @MapperScan
注解,扫描Mapper包下的所有接口: 1 2 3 4 5 6 7 8 9 @SpringBootApplication @MapperScan("com.github.mapper") public class Mybatisplus01Application { public static void main (String[] args) { SpringApplication.run(Mybatisplus01Application.class, args); } }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 @SpringBootTest class Mybatisplus01ApplicationTests { @Autowired private UserMapper userMapper; @Test void contextLoads () { List<User> users = userMapper.selectList(null ); users.forEach(System.out::println); } }
提示:UserMapper 中的 selectList()
方法的参数为 MP 内置的条件封装器 Wrapper
,所以不填写就是无任何条件。
小结一下:
通过以上几个简单的步骤,我们就实现了 User 表的 CRUD 功能,甚至连 XML 文件都不用编写! 从以上步骤中,我们可以看到集成MyBatis-Plus
非常的简单,只需要引入 starter 工程,并配置 mapper 扫描路径即可。 思考问题:
sql谁帮我们写的?—mybatis-plus 方法谁帮我们写的?—mybatis-plus 3.配置日志 所有的sql现在是不可见的,如果我想知道它是怎么执行的,所以我必须要看日志! 1 2 mybatis-plus.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl
配置完毕日志之后,后面的学习就需要注意这个自动生成的SQL,我们就会喜欢上 MyBatis-Plus! 4.CRUD 1.插入操作 1 2 3 4 5 6 7 8 9 10 11 12 13 14 @Test public void insertTest () { User user = new User (); user.setName("哇哈哈" ); user.setAge(22 ); user.setEmail("2589654784@qq.com" ); Integer insert = userMapper.insert(user); System.out.println(insert); System.out.println(user); }
2.主键生成策略 默认 ID_WORKER 全局唯一id
分布式系统唯一id生成:查看博客园 分布式id生成算法的有很多种,Twitter的SnowFlake就是其中经典的一种。 雪花算法 :
snowflake是Twitter开源的分布式ID生成算法,结果是一个long型的ID。==SnowFlake算法生成id的结果是一个64bit大小的整数==。 其核心思想是:使用41bit作为毫秒数,10bit作为机器的ID(5个bit是数据中心,5个bit的机器ID),12bit作为毫秒内的流水号(意味着每个节点在每毫秒可以产生 4096 个 ID),最后还有一个符号位,永远是0。可以保证几乎全球唯一!
算法实现 :
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 package com.twitter.service.snowflakeimport com.twitter.ostrich.stats.Statsimport com.twitter.service.snowflake.gen._import java.util.Randomimport com.twitter.logging.Loggerclass IdWorker (val workerId: Long, val datacenterId: Long, private val reporter: Reporter, var sequence: Long = 0L )extends Snowflake .Iface { private [this ] def genCounter (agent: String) = { Stats.incr("ids_generated" ) Stats.incr("ids_generated_%s" .format(agent)) } private [this ] val exceptionCounter = Stats.getCounter("exceptions" ) private [this ] val log = Logger.get private [this ] val rand = new Random val twepoch = 1288834974657L private [this ] val workerIdBits = 5L private [this ] val datacenterIdBits = 5L private [this ] val maxWorkerId = -1L ^ (-1L << workerIdBits) private [this ] val maxDatacenterId = -1L ^ (-1L << datacenterIdBits) private [this ] val sequenceBits = 12L private [this ] val workerIdShift = sequenceBits private [this ] val datacenterIdShift = sequenceBits + workerIdBits private [this ] val timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits private [this ] val sequenceMask = -1L ^ (-1L << sequenceBits) private [this ] var lastTimestamp = -1L if (workerId > maxWorkerId || workerId < 0 ) { exceptionCounter.incr(1 ) throw new IllegalArgumentException ("worker Id can't be greater than %d or less than 0" .format(maxWorkerId)) } if (datacenterId > maxDatacenterId || datacenterId < 0 ) { exceptionCounter.incr(1 ) throw new IllegalArgumentException ("datacenter Id can't be greater than %d or less than 0" .format(maxDatacenterId)) } log.info("worker starting. timestamp left shift %d, datacenter id bits %d, worker id bits %d, sequence bits %d, workerid %d" , timestampLeftShift, datacenterIdBits, workerIdBits, sequenceBits, workerId) def get_id (useragent: String) : Long = { if (!validUseragent(useragent)) { exceptionCounter.incr(1 ) throw new InvalidUserAgentError } val id = nextId() genCounter(useragent) reporter.report(new AuditLogEntry (id, useragent, rand.nextLong)) id } def get_worker_id () : Long = workerId def get_datacenter_id () : Long = datacenterId def get_timestamp () = System.currentTimeMillis protected [snowflake] def nextId () : Long = synchronized { var timestamp = timeGen() if (timestamp < lastTimestamp) { exceptionCounter.incr(1 ) log.error("clock is moving backwards. Rejecting requests until %d." , lastTimestamp); throw new InvalidSystemClock ("Clock moved backwards. Refusing to generate id for %d milliseconds" .format( lastTimestamp - timestamp)) } if (lastTimestamp == timestamp) { sequence = (sequence + 1 ) & sequenceMask if (sequence == 0 ) { timestamp = tilNextMillis(lastTimestamp) } } else { sequence = 0 } lastTimestamp = timestamp ((timestamp - twepoch) << timestampLeftShift) | (datacenterId << datacenterIdShift) | (workerId << workerIdShift) | sequence } protected def tilNextMillis (lastTimestamp: Long) : Long = { var timestamp = timeGen() while (timestamp <= lastTimestamp) { timestamp = timeGen() } timestamp } protected def timeGen () : Long = System.currentTimeMillis() val AgentParser = """([a-zA-Z][a-zA-Z\-0-9]*)""" .r def validUseragent (useragent: String) : Boolean = useragent match { case AgentParser (_) => true case _ = > false } }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 public class SnowflakeIdWorker { private final long twepoch = 1575365018000L ; private final long workerIdBits = 10L ; private final long maxWorkerId = -1L ^ (-1L << workerIdBits); private final long sequenceBits = 12L ; private final long workerIdShift = sequenceBits; private final long timestampLeftShift = sequenceBits + workerIdBits; private final long sequenceMask = -1L ^ (-1L << sequenceBits); private long workerId; private long sequence = 0L ; private long lastTimestamp = -1L ; public SnowflakeIdWorker (long workerId) { if (workerId > maxWorkerId || workerId < 0 ) { throw new IllegalArgumentException (String.format("workerId can't be greater than %d or less than 0" , maxWorkerId)); } this .workerId = workerId; } public synchronized long nextId () { long timestamp = timeGen(); if (timestamp < lastTimestamp) { throw new RuntimeException ( String.format("Clock moved backwards. Refusing to generate id for %d milliseconds" , lastTimestamp - timestamp)); } if (lastTimestamp == timestamp) { sequence = (sequence + 1 ) & sequenceMask; if (sequence == 0 ) { timestamp = tilNextMillis(lastTimestamp); } } else { sequence = 0L ; } lastTimestamp = timestamp; return ((timestamp - twepoch) << timestampLeftShift) | (workerId << workerIdShift) | sequence; } protected long tilNextMillis (long lastTimestamp) { long timestamp = timeGen(); while (timestamp <= lastTimestamp) { timestamp = timeGen(); } return timestamp; } protected long timeGen () { return System.currentTimeMillis(); } }
主键自增
实体类字段上 @TableId(type = IdType.AUTO)
数据库字段一定要是自增!
再次测试插入!
相关源码解释:
1 2 3 4 5 6 7 8 public enum IdType { AUTO(0 ), NONE(1 ), INPUT(2 ), ID_WORKER(3 ), UUID(4 ), ID_WORKER_STR(5 ); }
3.更新操作 1 2 3 4 5 6 7 8 9 10 11 12 @Test public void UpdateTest () { User user = new User (); user.setId(7L ); user.setName("KYDH,开源导航" ); user.setAge(26 ); Integer i = userMapper.updateById(user); System.out.println(i); }
4.自动填充 创建时间、更改时间! 这些操作一般都是自动化完成,我们不希望手动更新。
阿里巴巴开发手册︰几乎所有的表都要配置 gmt_create、gmt_modified !而且需要自动化。
方式一:数据库级别(工作中不允许修改数据库级别)
在表中增加字段:create_time,update_time
再次测试插入或更新方法,需要在实体类中同步! 1 2 private Date createTime; private Date updateTime;
查看结果
方式二:代码级别
删除数据库的默认值,更新操作!
实体类字段属性上需要增加注解 1 2 3 4 @TableField(fill = FieldFill.INSERT) private Date createTime; @TableField(fill = FieldFill.INSERT_UPDATE) private Date updateTime;
编写处理器来处理这个注解即可! 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 package com.github.handler;import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler;import lombok.extern.slf4j.Slf4j;import org.apache.ibatis.reflection.MetaObject;import org.springframework.stereotype.Component;@Slf4j @Component public class MyMetaObjectHandler implements MetaObjectHandler { @Override public void insertFill (MetaObject metaObject) { log.info("start insert fill......" ); this .setFieldValByName("createTime" ,new Date (),metaObject); this .setFieldValByName("updateTime" ,new Date (),metaObject); } @Override public void updateFill (MetaObject metaObject) { log.info("start update fill……" ); this .setFieldValByName("updateTime" ,new Date (),metaObject); } }
测试插入/更新,观察时间
5.乐观锁和悲观锁 乐观锁:顾名思义十分乐观,他总是认为不会出现问题,无论干什么都不上锁!如果出现了问题,再次更新值测试。
悲观锁:顾名思义十分悲观,他总是认为出现问题,无论干什么都会上锁!再去操作!
这里主要讲解:乐观锁机制 !
乐观锁实现方式:
取出记录时,获取当前version 更新时,带上这个version 执行更新时,set version = newVersion where version = oldVersion 如果version不对,就更新失败 1 2 3 4 5 6 7 乐观锁:1 、先查询,获得版本号 version = 1 update user set name = "哇哈哈", version = version + 1 where id = 2 and version = 1 update user set name = "哇哈哈", version = version + 1 where id = 2 and version = 1
测试一下Mybatis-Plus乐观锁插件
给数据库中增加version字段
实体类加对应的字段 1 2 @Version private Integer version;
注册组件 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 package com.github.config;import com.baomidou.mybatisplus.extension.plugins.OptimisticLockerInterceptor;import org.mybatis.spring.annotation.MapperScan;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.transaction.annotation.EnableTransactionManagement;@MapperScan("com.github.mapper") @EnableTransactionManagement @Configuration public class MyBatisPlusConfig { @Bean public OptimisticLockerInterceptor optimisticLockerInterceptor () { return new OptimisticLockerInterceptor (); } }
测试一下: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 @Test public void testOptimisticLocker () { User user = userMapper.selectById(1L ); user.setName("金顶" ); user.setEmail("2451367@qq.com" ); userMapper.updateById(user); } @Test public void testOptimisticLocker2 () { User user = userMapper.selectById(1L ); user.setName("夸父1" ); user.setEmail("2451367@qq.com" ); User user2 = userMapper.selectById(1L ); user2.setName("夸父2" ); user2.setEmail("2451367@qq.com" ); userMapper.updateById(user2); userMapper.updateById(user); }
6.查询操作 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 @Test public void SelectByIdTest () { User user = userMapper.selectById(1L ); System.out.println(user); } @Test public void SelectByBatchIdTest () { List<User> users = userMapper.selectBatchIds(Arrays.asList(1 , 2 , 3 )); users.forEach(System.out::println); } @Test public void SelectByBatchIds () { HashMap<String, Object> map = new HashMap <>(); map.put("name" ,"哇哈哈" ); map.put("age" ,22 ); List<User> users = userMapper.selectByMap(map); users.forEach(System.out::println); }
7.分页查询 分页在网站使用的十分之多!
原始的 limit 进行分页 pageHelper 第三方插件 MP 其实也内置了分页插件! 具体使用:
配置拦截器组件即可! 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 @MapperScan("com.github.mapper") @EnableTransactionManagement @Configuration public class MyBatisPlusConfig { @Bean public OptimisticLockerInterceptor optimisticLockerInterceptor () { return new OptimisticLockerInterceptor (); } @Bean public PaginationInterceptor paginationInterceptor () { return new PaginationInterceptor (); } }
直接使用Page对象即可! 1 2 3 4 5 6 7 8 9 10 11 @Test public void testPage () { Page<User> page = new Page <>(2 ,5 ); userMapper.selectPage(page,null ); page.getRecords().forEach(System.out::println); System.out.println("总页数==>" +page.getTotal()); }
8.删除操作 根据 id 删除记录
1 2 3 4 5 6 7 8 9 10 11 @Test public void DeleteTest () { userMapper.deleteById(14993819200198L ); } @Test public void DeleteByIdTest () { userMapper.deleteBatchIds(Arrays.asList(14993819200199L ,6 )); }
通过map删除
1 2 3 4 5 6 7 @Test public void DeleteMapTest () { HashMap<String, Object> map = new HashMap <>(); map.put("name" ,"KYDH,开源导航" ); userMapper.deleteByMap(map); }
9.逻辑删除 物理删除 :从数据库中直接移除;
逻辑删除 :再数据库中没有被移除,而是通过一个变量来让他失效! deleted = 0 => deleted = 1;
管理员可以查看被删除的记录!防止数据的丢失,类似于回收站! 测试一下 :
在数据表中增加一个 deleted 字段
实体类中增加属性 1 2 @TableLogic private Integer deleted;
配置 1 2 3 4 5 @Bean public ISqlInjector sqlInjector () { return new LogicSqlInjector (); }
1 2 3 mybatis-plus.global-config.db-config.logic-delete-value=1 mybatis-plus.global-config.db-config.logic-not-delete-value=0
4.测试!
走的是更新,不是删除操作!记录依旧在数据库,但是值确已经变化了!
==以上的所有CRUD操作及其扩展操作,都必须精通掌握==!会大大提高你的工作和写项目的效率! 5.性能分析插件 在平时的开发中,会遇到一些慢sql。测试! druid,,,,, 作用:性能分析拦截器,用于输出每条 SQL 语句及其执行时间 MP也提供性能分析插件,如果超过这个时间就停止运行! 导入插件 1 2 3 4 5 6 7 8 9 10 11 @Bean @Profile({"dev","test"}) public PerformanceInterceptor performanceInterceptor () { PerformanceInterceptor performanceInterceptor = new PerformanceInterceptor (); performanceInterceptor.setMaxTime(100 ); performanceInterceptor.setFormat(true ); return performanceInterceptor; }
注意:要在SpringBoot中配置环境为dev或者 test 环境! 1 2 spring.profiles.active =dev
测试一下! 1 2 3 4 5 6 7 @Test void contextLoads () { List<User> users = userMapper.selectList(null ); users.forEach(System.out::println); }
6.条件构造器Wrapper
测试一
1 2 3 4 5 6 7 8 9 10 @Test void contextLoads2 () { QueryWrapper<User> wrapper = new QueryWrapper <>(); wrapper .isNotNull("name" ) .isNotNull("email" ) .ge("age" ,12 ); userMapper.selectList(wrapper).forEach(System.out::println); }
测试二
1 2 3 4 5 6 7 8 @Test void test2 () { QueryWrapper<User> wrapper = new QueryWrapper <>(); wrapper.eq("name" ,"Jack" ); User user = userMapper.selectOne(wrapper); System.out.println(user); }
测试三
1 2 3 4 5 6 7 8 @Test void test3 () { QueryWrapper<User> wrapper = new QueryWrapper <>(); wrapper.between("age" ,20 ,30 ); Integer count = userMapper.selectCount(wrapper); System.out.println(count); }
测试四
1 2 3 4 5 6 7 8 9 10 11 12 @Test void test4 () { QueryWrapper<User> wrapper = new QueryWrapper <>(); wrapper .notLike("name" ,"e" ) .likeRight("email" ,"t" ); List<Map<String, Object>> maps = userMapper.selectMaps(wrapper); maps.forEach(System.out::println); }
测试五()
1 2 3 4 5 6 7 8 9 10 11 12 13 @Test public void testWrapper5 () { QueryWrapper<User> wrapper = new QueryWrapper <>(); wrapper.inSql("id" ,"select id from user where id<5" ); List<Object> objects = userMapper.selectObjs(wrapper); objects.forEach(System.out::println); }
测试六
1 2 3 4 5 6 7 8 @Test public void testWrapper6 () { QueryWrapper<User> wrapper = new QueryWrapper <>(); wrapper.orderByDesc("id" ); List<User> userList = userMapper.selectList(wrapper); userList.forEach(System.out::println); }
7.代码自动生成器 AutoGenerator
是 MyBatis-Plus 的代码生成器,通过 AutoGenerator 可以快速生成 Entity、Mapper、Mapper XML、Service、Controller等各个模块的代码,极大的提升了开发效率。旧版测试:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 package com.github;import com.baomidou.mybatisplus.annotation.DbType;import com.baomidou.mybatisplus.annotation.FieldFill;import com.baomidou.mybatisplus.annotation.IdType;import com.baomidou.mybatisplus.generator.AutoGenerator;import com.baomidou.mybatisplus.generator.config.DataSourceConfig;import com.baomidou.mybatisplus.generator.config.GlobalConfig;import com.baomidou.mybatisplus.generator.config.PackageConfig;import com.baomidou.mybatisplus.generator.config.StrategyConfig;import com.baomidou.mybatisplus.generator.config.po.TableFill;import com.baomidou.mybatisplus.generator.config.rules.DateType;import com.baomidou.mybatisplus.generator.config.rules.NamingStrategy;import java.util.ArrayList;public class WskCode { public static void main (String[] args) { AutoGenerator mpg = new AutoGenerator (); GlobalConfig gc = new GlobalConfig (); String projectPath = System.getProperty("user.dir" ); gc.setOutputDir(projectPath+"/src/main/java" ); gc.setAuthor("github" ); gc.setOpen(false ); gc.setFileOverride(false ); gc.setServiceName("%sService" ); gc.setIdType(IdType.ID_WORKER); gc.setDateType(DateType.ONLY_DATE); gc.setSwagger2(true ); mpg.setGlobalConfig(gc); DataSourceConfig dsc = new DataSourceConfig (); dsc.setUsername("root" ); dsc.setPassword("root" ); dsc.setUrl("jdbc:mysql://localhost:3306/github?useSSL=false&serverTimezone=GMT%2B8&useUnicode=true&characterEncoding=utf-8" ); dsc.setDriverName("com.mysql.cj.jdbc.Driver" ); dsc.setDbType(DbType.MYSQL); mpg.setDataSource(dsc); PackageConfig pc = new PackageConfig (); pc.setModuleName("study" ); pc.setParent("com.github" ); pc.setEntity("pojo" ); pc.setMapper("mapper" ); pc.setService("service" ); pc.setController("controller" ); mpg.setPackageInfo(pc); StrategyConfig strategy = new StrategyConfig (); strategy.setInclude("admin" ,"Banyan" ,"building" ,"room" ); strategy.setNaming(NamingStrategy.underline_to_camel); strategy.setColumnNaming(NamingStrategy.underline_to_camel); strategy.setEntityLombokModel(true ); strategy.setLogicDeleteFieldName("deleted" ); TableFill gmtCreate = new TableFill ("gmt_create" , FieldFill.INSERT); TableFill gmtUpdate = new TableFill ("gmt_update" , FieldFill.INSERT_UPDATE); ArrayList<TableFill> tableFills = new ArrayList <>(); tableFills.add(gmtCreate); tableFills.add(gmtUpdate); strategy.setTableFillList(tableFills); strategy.setVersionFieldName("version" ); strategy.setRestControllerStyle(true ); strategy.setControllerMappingHyphenStyle(true ); mpg.setStrategy(strategy); mpg.execute(); } }
新版测试
1 2 3 4 5 6 7 8 9 10 <dependency > <groupId > com.baomidou</groupId > <artifactId > mybatis-plus-generator</artifactId > <version > 3.5.1</version > </dependency > <dependency > <groupId > org.freemarker</groupId > <artifactId > freemarker</artifactId > <version > 2.3.31</version > </dependency >
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 package com.github;import com.baomidou.mybatisplus.generator.FastAutoGenerator;import com.baomidou.mybatisplus.generator.config.OutputFile;import com.baomidou.mybatisplus.generator.engine.FreemarkerTemplateEngine;import java.util.Collections;public class QuaryCode { public static void main (String[] args) { FastAutoGenerator.create("url" , "username" , "password" ) .globalConfig(builder -> { builder.author("test" ) .enableSwagger() .fileOverride() .outputDir("D://" ); }) .packageConfig(builder -> { builder.parent("com.github.mybatisplus.samples.generator" ) .moduleName("system" ) .pathInfo(Collections.singletonMap(OutputFile.mapperXml, "D://" )); }) .strategyConfig(builder -> { builder.addInclude("t_simple" ) .addTablePrefix("t_" , "c_" ); }) .templateEngine(new FreemarkerTemplateEngine ()) .execute(); } }
8.多数据源 适用于多种场景:纯粹多库、 读写分离、 一主多从、混合模式等。 目前我们就来模拟一个纯粹多库的一个场景,其他场景类似。 场景说明:创建两个库,分别为:mybatis_plus(以前的库不动)与mybatis_plus_1(新建); 将mybatis_plus库的product表移动到mybatis_plus_1库,这样每个库一张表; 通过一个测试用例分别获取用户数据与商品数据,如果获取到说明多库模拟成功。 创建数据库mybatis_plus_1和表product
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 CREATE DATABASE `mybatis_plus` /*!40100 DEFAULT CHARACTER SET utf8mb4 */; use `mybatis_plus`; CREATE TABLE `user` ( `id` bigint(20) NOT NULL COMMENT '主键ID', `name` varchar(30) DEFAULT NULL COMMENT '姓名', `age` int(11) DEFAULT NULL COMMENT '年龄', `email` varchar(50) DEFAULT NULL COMMENT '邮箱', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8; 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');
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 CREATE DATABASE `mybatis_plus_1` /*!40100 DEFAULT CHARACTER SET utf8mb4 */; use `mybatis_plus_1`; CREATE TABLE product ( id BIGINT(20) NOT NULL COMMENT '主键ID', name VARCHAR(30) NULL DEFAULT NULL COMMENT '商品名称', price INT(11) DEFAULT 0 COMMENT '价格', version INT(11) DEFAULT 0 COMMENT '乐观锁版本号', PRIMARY KEY (id) ); # 添加测试数据 INSERT INTO product (id, NAME, price) VALUES (1, '外星人笔记本', 100); # 删除mybatis_plus库product表 use mybatis_plus; DROP TABLE IF EXISTS product;
新建一个spring boot项目,引入依赖:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 <dependencies > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-web</artifactId > </dependency > <dependency > <groupId > com.baomidou</groupId > <artifactId > mybatis-plus-boot-starter</artifactId > <version > 3.5.1</version > </dependency > <dependency > <groupId > mysql</groupId > <artifactId > mysql-connector-java</artifactId > <scope > runtime</scope > </dependency > <dependency > <groupId > com.baomidou</groupId > <artifactId > dynamic-datasource-spring-boot-starter</artifactId > <version > 3.5.0</version > </dependency > <dependency > <groupId > org.projectlombok</groupId > <artifactId > lombok</artifactId > <optional > true</optional > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-test</artifactId > <scope > test</scope > </dependency > </dependencies >
配置多数据源
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 spring: datasource: dynamic: primary: master strict: false datasource: master: url: jdbc:mysql://localhost:3306/mybatis_plus?characterEncoding=utf-8&useSSL=false driver-class-name: com.mysql.cj.jdbc.Driver username: root password: root slave_1: url: jdbc:mysql://localhost:3306/mybatis_plus_1?characterEncoding=utf-8&useSSL=false driver-class-name: com.mysql.cj.jdbc.Driver username: root password: root
创建实体类
1 2 3 4 5 6 7 8 9 @Data @TableName("user") public class User { @TableField private Long id; private String name; private Integer age; private String email; }
1 2 3 4 5 6 7 @Data public class Product { private Integer id; private String name; private Integer price; private Integer version; }
创建mapper
1 2 3 @Repository public interface UserMapper extends BaseMapper <User> {}
1 2 3 @Repository public interface ProductMapper extends BaseMapper <Product> {}
启动类
1 2 3 4 5 6 7 8 9 @SpringBootApplication @MapperScan("com.github.mapper") public class MybatisPlusDatasourceApplication { public static void main (String[] args) { SpringApplication.run(MybatisPlusDatasourceApplication.class, args); } }
创建用户service
1 2 public interface UserService extends IService <User> {}
1 2 3 4 @DS("master") @Service public class UserServiceImpl extends ServiceImpl <UserMapper, User> implements UserService {}
创建商品service
1 2 public interface ProductService extends IService <Product> {}
1 2 3 4 @DS("slave_1") @Service public class ProductServiceImpl extends ServiceImpl <ProductMapper, Product> implements ProductService {}
测试
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 @SpringBootTest class MybatisPlusDatasourceApplicationTests { @Autowired private UserService userService; @Autowired private ProductService productService; @Test public void testDynamicDataSource () { System.out.println(userService.getById(1L )); System.out.println(productService.getById(1L )); } }
9.MyBatisX插件 MyBatis-Plus为我们提供了强大的mapper和service模板,能够大大的提高开发效率。但是在真正开发过程中,MyBatis-Plus并不能为我们解决所有问题,例如一些复杂的SQL,多表联查,我们就需要自己去编写代码和SQL语句,我们该如何快速的解决这个问题呢,这个时候可以使用MyBatisX插件MyBatisX一款基于 IDEA 的快速开发插件,为效率而生。
安装方法:打开 IDEA,进入 File -> Settings -> Plugins -> Browse Repositories,输入 mybatisx
搜索并安装。
功能 :
JPA 提示
🎉结束了🎉