目录
简介
安装
配置
实体类常用注解
使用
CRUD接口
条件构造器
分页
MyBatis-plus是mybatis的增强工具,在MyBatis 上只做增强,不做改变,引入他不会对现有工程产生影响,只需简单配置快速实现CURD操作,从而节省大量时间。代码生成、自动分页、j逻辑删除、自动、填充等功能一应俱全。
首先创建一个springboot项目,导入Mybatis-plus相关依赖:
mysql
mysql-connector-java
8.0.21
com.alibaba
druid-spring-boot-starter
1.1.10
com.baomidou
mybatis-plus-boot-starter
3.4.2
由于Mybatis-plus是包含Mybatis 依赖的所以导入Mybatis-plus后就不用导入Mybatis了
在springboot的yml文件中可以配置实体类别名和mapper文件路径:
mybatis-plus:
mapper-locations: classpath*:/mapper/**/*.xml
type-aliases-package: com.xxz.pojo
mapperlocations 自动配置好的,默认值是classpath*:/mapper/**/*.xml 意为任意包路径下所有的mapper包下的xml文件
也可以使用Mybatis中的方法进行配置,因为Mybatis-plus是只做增强不做改变。
@TableName("dept") 对应数据库的表明
@TableField("deptno") 对应数据库的字段名
@TableField(exist = false) 是否为数据库表字段
详细文档:https://baomidou.com/pages/223848/
@AllArgsConstructor
@NoArgsConstructor
@Data
@TableName("dept")//用于数据库表明和实体类名不一致情况
public class Dept implements Serializable {
/*@TableField(exist = false)
private String aaa;*/
@TableField("deptno")//用于数据库字段名和实体类属性不一致情况
private Integer deptno;
private String dname;
private String loc;
}
mapper类需要实现BaseMapper
接口,传入实体类的泛型
public interface DeptMapper extends BaseMapper {
}
service接口需要继承Iservice
类,传入实体类的泛型
public interface DeptService extends IService {
}
service实现类需要继承ServiceImpl
类,传入mapper类,和实体类的泛型
@Service
public class DeptServiceImpl extends ServiceImpl implements DeptService {
}
save方法:
// 插入一条记录(选择字段,策略插入)
boolean save(T entity); entity实体对象
// 插入(批量)
boolean saveBatch(CollectionentityList); entityList实体对象集合
// 插入(批量)
boolean saveBatch(CollectionentityList, int batchSize); batchSize插入批次数量
// 增加
@Test
public void testAdd(){
boolean save = deptService.save(new Dept(null, "aaa", "bbb"));
System.out.println(save);
}
list方法:
// 查询所有
Listlist();
// 查询列表
Listlist(Wrapper queryWrapper);queryWrapper条件构造
// 查询集合
@Test
public void testQueryWrapper(){
// 部门号》=20
// QueryWrapper 作用就是在原本的SQL语句后面拼接where条件
// selec * from where delete from dept where update dept set ... where ....
QueryWrapper queryWrapper=new QueryWrapper<>();
//queryWrapper.ge("deptno", 20).eq("dname", "ACCOUNTING").likeRight("dname", "A");
//queryWrapper.likeRight("dname", "A");
List list = deptService.list(queryWrapper);
for (Dept dept : list) {
System.out.println(dept);
}
}
// 查询单个
@Test
public void testQueryWrapper2(){
QueryWrapper queryWrapper=new QueryWrapper<>();
queryWrapper.eq("deptno", 20);
Dept dept = deptService.getOne(queryWrapper);
System.out.println(dept);
}
update方法:
// 根据 UpdateWrapper 条件
boolean update(WrapperupdateWrapper);
// 根据 whereWrapper 条件,更新记录
boolean update(T updateEntity, WrapperwhereWrapper);
// 根据 ID 选择修改
boolean updateById(T entity);
// 根据ID 批量更新
boolean updateBatchById(CollectionentityList);
// 根据ID 批量更新
boolean updateBatchById(CollectionentityList, int batchSize);
// 修改
@Test
public void testUpdate(){
// 要更新的数据
Dept dept =new Dept();
dept.setDname("xxx");
dept.setLoc("yyy");
// 更新的条件
QueryWrapper queryWrapper=new QueryWrapper<>();
queryWrapper.eq("deptno", 41);
boolean update = deptService.update(dept, queryWrapper);
System.out.println(update);
}
remove方法:
// 根据 entity 条件,删除记录
boolean remove(WrapperqueryWrapper);
// 删除
@Test
public void testRemove(){
QueryWrapper queryWrapper=new QueryWrapper<>();
queryWrapper.eq("deptno", 41);
boolean remove = deptService.remove(queryWrapper);
System.out.println(remove);
}
详细文档:https://baomidou.com/pages/49cc81/
AbstractWrapper
说明: QueryWrapper(LambdaQueryWrapper) 和 UpdateWrapper(LambdaUpdateWrapper) 的父类
用于生成 sql 的 where 条件, entity 属性也用于生成 sql 的 where 条件
注意: entity 生成的 where 条件与 使用各个 api 生成的 where 条件没有任何关联行为
方法 :
eq("name", "老王")
--->name = '老王'
ne("name", "老王")
--->name <> '老王'
gt("age", 18)
--->age > 18
ge("age", 18)
--->age >= 18
lt("age", 18)
--->age < 18
le("age", 18)
--->age <= 18
between("age", 18, 30)
--->age between 18 and 30
notBetween("age", 18, 30)
--->age not between 18 and 30
like("name", "王")
--->name like '%王%'
notLike("name", "王")
--->name not like '%王%'
likeLeft("name", "王")
--->name like '%王'
likeRight("name", "王")
--->name like '王%'
isNull("name")
--->name is null
isNotNull("name")
--->name is not null
in("age",{1,2,3})
--->age in (1,2,3)
notIn("age",{1,2,3})
--->age not in (1,2,3)
inSql("age", "1,2,3,4,5,6")
--->age in (1,2,3,4,5,6)
例: inSql("id", "select id from table where id < 3")
--->id in (select id from table where id < 3)
notInSql("age", "1,2,3,4,5,6")
--->age not in (1,2,3,4,5,6)
例: notInSql("id", "select id from table where id < 3")
--->id not in (select id from table where id < 3)
groupBy("id", "name")
--->group by id,name
orderByAsc("id", "name")
--->order by id ASC,name ASC
orderByDesc("id", "name")
--->order by id DESC,name DESC
orderBy(true, true, "id", "name")
--->order by id ASC,name ASC
having("sum(age) > 10")
--->having sum(age) > 10
例: having("sum(age) > {0}", 11)
--->having sum(age) > 11
or:
拼接 OR
- 例:
eq("id",1).or().eq("name","老王")
--->id = 1 or name = '老王'
注意事项: 主动调用or表示紧接着下一个方法不是用and连接!(不调用or则默认为使用and连接)
OR 嵌套
- 例:
or(i -> i.eq("name", "李白").ne("status", "活着"))
--->or (name = '李白' and status <> '活着')
and:
AND 嵌套
- 例:
and(i -> i.eq("name", "李白").ne("status", "活着"))
--->and (name = '李白' and status <> '活着')
nested:
正常嵌套 不带 AND 或者 OR
- 例:
nested(i -> i.eq("name", "李白").ne("status", "活着"))
--->(name = '李白' and status <> '活着')
apply:
拼接 sql
注意事项:
该方法可用于数据库函数 动态入参的
params
对应前面applySql
内部的{index}
部分.这样是不会有sql注入风险的,反之会有!
- 例:
apply("id = 1")
--->id = 1
- 例:
apply("date_format(dateColumn,'%Y-%m-%d') = '2008-08-08'")
--->date_format(dateColumn,'%Y-%m-%d') = '2008-08-08'")
- 例:
apply("date_format(dateColumn,'%Y-%m-%d') = {0}", "2008-08-08")
--->date_format(dateColumn,'%Y-%m-%d') = '2008-08-08'")
last:
无视优化规则直接拼接到 sql 的最后
注意事项:
只能调用一次,多次调用以最后一次为准 有sql注入的风险,请谨慎使用
- 例:
last("limit 1")
exists:
拼接 EXISTS ( sql语句 )
- 例:
exists("select id from table where age = 1")
--->exists (select id from table where age = 1)
notExists:
拼接 NOT EXISTS ( sql语句 )
- 例:
notExists("select id from table where age = 1")
--->not exists (select id from table where age = 1)
select:
设置查询字段
select(String... sqlSelect)
select(Predicatepredicate)
select(ClassentityClass, Predicate predicate) 说明:
以上方法分为两类.
第二类方法为:过滤查询字段(主键除外),入参不包含 class 的调用前需要wrapper
内的entity
属性有值! 这两类方法重复调用以最后一次为准
- 例:
select("id", "name", "age")
- 例:
select(i -> i.getProperty().startsWith("test"))
UpdateWrapper :
set:
SQL SET 字段
- 例:
set("name", "老李头")
- 例:
set("name", "")
--->数据库字段值变为空字符串- 例:
set("name", null)
--->数据库字段值变为null
setSql:
设置 SET 部分 SQL
- 例:
setSql("name = '老李头'")
配置分页插件
@Configuration
@MapperScan("scan.your.mapper.package")
public class MybatisPlusConfig {
/**
* 新的分页插件,一缓和二缓遵循mybatis的规则,需要设置 MybatisConfiguration#useDeprecatedExecutor = false 避免缓存出现问题(该属性会在旧插件移除后一同移除)
*/
@Bean
public MybatisPlusInterceptor mybatisPlusInterceptor() {
MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.H2));
return interceptor;
}
@Bean
public ConfigurationCustomizer configurationCustomizer() {
return configuration -> configuration.setUseDeprecatedExecutor(false);
}
}
测试分页插件
@Test
public void testPage(){
// 当前页 页大小
QueryWrapper queryWrapper=new QueryWrapper<>();
//queryWrapper.likeRight("dname", "A");
Page page = deptService.page(new Page<>(1, 2), queryWrapper);
// 当前页数据 总页数 总记录数 当前页 页大小 ... ..
List list = page.getRecords();
list.forEach(System.out::println);
System.out.println("总页数:"+page.getPages());
System.out.println("总记录数:"+page.getTotal());
System.out.println("当前页:"+page.getCurrent());
System.out.println("页大小:"+page.getSize());
}