前言
上节为通过id、idList、idSet、Map设置条件进行等值查询,但在实际使用中,还需要使用> <等连接符进行查询,那么此次则介绍条件构造器这一使用方法。
这个是自己学习时候记得笔记要是想详细了解可以去MP官网,上边有更详细的配置流程以及视频教学:MyBatis-Plus
条件构造器使用(QueryWrapper)
- 执行单表操作的 like > < is not Null等方法
public List<Type> selectByWrapper() {
// 创建方式1. new
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
// 创建方式2. Wappers.query()
QueryWrapper<Type> queryWrapper1 = Wrappers.<Type>query();
// 构造条件 : 单表查询使用and连接
queryWrapper.like("name", "a") // like 模糊查询
.lt("parentId", 5) // lt: lessThan 小于
.isNotNull("id"); // 不为空
// 以上条件均and进行连接;
// 构造条件: 单表查询使用or连接
queryWrapper1.likeRight("name", "王") // 姓王, 王%
.or() // 使用 or 进行连接
.ge("parentId", 3) // ge : great Than , 大于
.orderByDesc("id") // 根据 id 降序
.orderByAsc("parentId"); // 根据 parentId升序
// 传入参数
return this.typeMapper.selectList(queryWrapper);
}
- 查询条件中含有 in 子句
public List<Type> selectByWrapperIn() {
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
queryWrapper.apply("date_format(productDate,'%Y-%m-%d)={0}", "2011-05-09")
// 添加条件,{0}为选择后面参数第0个位置,形成SQL使用?占位符,防止SQL注入
.inSql("type_id", "select type_id from type where name like '生%'");
// 添加 in 语句
return this.typeMapper.selectList(queryWrapper);
}
- 同时含有and or:先and后or,且使用() 将 or 的优先级提高 [a and (b or c)]
public List<Type> selectByWrapperAndOr() {
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
queryWrapper.likeRight("name", "生") // like '生%'
.and(new Consumer<QueryWrapper<Type>>() {
@Override
public void accept(QueryWrapper<Type> typeQueryWrapper) {
typeQueryWrapper.lt("parentId", 2)
.or()
.isNotNull("productDate");
}
}); // and ( parentId = ? or productDate is not null)
return this.typeMapper.selectList(queryWrapper);
}
- 同时含有and or:先or后and,且使用() 将 and 的优先级提高 [ a or (b and c)]
public List<Type> selectByWrapperOrAnd() {
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
queryWrapper.likeRight("name", "生") // name like '生%'
.or(new Consumer<QueryWrapper<Type>>() {
@Override
public void accept(QueryWrapper<Type> typeQueryWrapper) {
typeQueryWrapper.lt("parentId", 20).gt("parentId", 0)
.isNotNull("productDate");
} // or (parentId < 20 and parentId > 0 and productDate is not null)
});
return this.typeMapper.selectList(queryWrapper);
}
- 同时含有and or:先or后and,使用()将or的优先级提高 [(a or b) and c]
public List<Type> selectByWrapperOrAnd1() {
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
queryWrapper.nested(new Consumer<QueryWrapper<Type>>() {
@Override
public void accept(QueryWrapper<Type> typeQueryWrapper) {
typeQueryWrapper.lt("parentId", 5)
.or().isNotNull("productDate");
} // (parentId < 5 or productDate is not null)
}).likeRight("name", "生"); // and name like '生%'
return this.typeMapper.selectList(queryWrapper);
}
- in语句,in(2,3,4)
public List<Type> selectByWrapperInCol() {
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
queryWrapper.in("parentId", Arrays.asList(2, 3, 4));
return this.typeMapper.selectList(queryWrapper);
}
- 只需要返回结果中满足条件的一条 (limit 1) (存在SQL注入风险)
public List<Type> selectByWrapperLimit() {
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
queryWrapper.in("parentId", Arrays.asList(2, 3, 4))
.last("limit 1");// .last("") 有sql注入风险
return this.typeMapper.selectList(queryWrapper);
}
- 指定返回字段
public List<Type> selectByWrapperSuperIdName() {
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
// 指定查询 方法1
queryWrapper.select("type_id", "name") // 只查询 type_id , name
.in("parentId", Arrays.asList(2, 3, 4))
.last("limit 1");// .last("") 有sql注入风险
// 指定查询 方法2
queryWrapper.select(Type.class, new Predicate<TableFieldInfo>() {
@Override
public boolean test(TableFieldInfo tableFieldInfo) {
return !tableFieldInfo.getColumn().equals("productDate")
&&
!tableFieldInfo.getColumn().equals("parentId");
}
}) // 得到查询结果时,排除 productDate 与 parentId 两条记录
.in("parentId", Arrays.asList(2, 3, 4))
.last("limit 1");// .last("") 有sql注入风险
return this.typeMapper.selectList(queryWrapper);
}
- condition的作用
在构造每个条件时,都会执行重载的方法,该方法第一位就是boolean condition
当condition 为 true 时,将条件组合进入 sql
需要判断非空时,可手动选择是否使用condition,若为空,则传入false
相当于MyBatis中动态SQL的
/**
* 条件构造器:condition 作用,
*/
public List<Type> selectByWrapperCondition(String name, int parentid) {
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
// 使用condition前
if (StringUtils.isNotEmpty(name)) {
queryWrapper.like("name", name);
}
if (parentid > 0) {
queryWrapper.eq("parentId", parentid);
}
// 使用condition后
queryWrapper.like(StringUtils.isNotEmpty(name), "name", name)
.eq(parentid > 0, "parentId", parentid);
return this.typeMapper.selectList(queryWrapper);
}
- 构造Wrapper时将实体作为参数:自动将entity中设置的属性放入查询条件
public List<Type> selectByWrapperEntity(){
Type type = new Type();
type.setName("abc");
QueryWrapper<Type> queryWrapper = new QueryWrapper<>(type);
// 注意1:重复设置name: 此时一个sql里边查询了两次name,用 and 连接
// queryWrapper.eq("name","abc");
// 注意2:实体类中在@TableField中配置condition属性,可控制在sql中使用什么条件
//@TableField(condition = SqlCondition.EQUAL)
//private int parentId;
/**
* public static final String EQUAL = "%s=#{%s}";
* public static final String NOT_EQUAL = "%s<>#{%s}";
* public static final String LIKE = "%s LIKE CONCAT('%%',#{%s},'%%')";
* public static final String LIKE_LEFT = "%s LIKE CONCAT('%%',#{%s})";
* public static final String LIKE_RIGHT = "%s LIKE CONCAT(#{%s},'%%')";
*/
// 可自行构造condition常量 : 大于 condition = "%s>#{%s}"
return this.typeMapper.selectList(queryWrapper);
}
- 条件构造器:AllEq 用法
allEq(Map)
allEq(Map,boolean isNull)
allEq(boolean condition,Map,isNull)
Map中,key : 数据库字段名,value : 字段值
isNull : true 在key对应value为null时,调用 key is null;
false 在key对应value对应为null时,自动忽略
allEq(BiPredicate,Map) 过滤指定字段让其加入 / 不加入sql语句
public List<Type> selectByWrapperAllEq(){
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
// Map<String,Object> map = new HashMap<>();
// map.put("name","生日");
// map.put("publishDate",null);
// queryWrapper.allEq(map); // allEq方法只有一个参数时,默认 isNull 为 true
Map<String,Object> map = new HashMap<>();
map.put("name","生日");
map.put("publishDate","2011-05-09");
queryWrapper.allEq(new BiPredicate<String, Object>() {
@Override
public boolean test(String s, Object o) {
return !s.equals("publishDate");
}
},map); // 指定publishDate不进入sql
return this.typeMapper.selectList(queryWrapper);
}
- 其他查询方法:selectMaps
返回 List< Map > 类型: ①当需要返回值较少,不需要封装成entity时使用
②返回统计结果时 (Count(id) ,avg、min、max)
public List<Map<String,Object>> selectWrapperMaps(){
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
// 情况1
queryWrapper.select("id","name").eq("name","生日");
// 情况2
queryWrapper.select("avg(id) avgId")
.groupBy("parentId")
.having("avgId > 2");
return this.typeMapper.selectMaps(queryWrapper);
}
- 其他查询方法:selectObjs
返回值List< Object >类型,返回值虽然是List,但是返回的数据只有一条
public List<Object> selectWrapperObjs(){
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
queryWrapper.select("type_id","name")
.like("name","巧克力")
.lt("parentId",20);
return this.typeMapper.selectObjs(queryWrapper);
}
- 其他查询方法:selectCount
public Integer selectWrapperCount(){
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
queryWrapper.like("name","巧克力")
.lt("parentId",20);
return this.typeMapper.selectCount(queryWrapper);
}
- 其他查询方法:selectOne
只查询/返回一条数据,多了会报错
预期返回一条数据或者null
public Type selectWrapperOne(){
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
queryWrapper.like("name","巧克力")
.lt("parentId",20);
return this.typeMapper.selectOne(queryWrapper);
}
- ambda 条件构造器l:可以有效防止查询字段误写
public List<Type> selectLambda(){
// 三种构造方式
// LambdaQueryWrapper<Type> lambda = new QueryWrapper<Type>().lambda();
// LambdaQueryWrapper<Type> lambdaQuery = new LambdaQueryWrapper<>();
LambdaQueryWrapper<Type> lambdaQueryWrapper = Wrappers.<Type>lambdaQuery();
// 可以有效防止查询字段误写
// lambdaQueryWrapper.like(Type::getName,'巧');// ==> where name like '%雨%'
// a and (b or c)
lambdaQueryWrapper.likeRight(Type::getName,"巧").and(new Consumer<LambdaQueryWrapper<Type>>() {
@Override
public void accept(LambdaQueryWrapper<Type> typeLambdaQueryWrapper) {
typeLambdaQueryWrapper.lt(Type::getParentId,5)
.or().ge(Type::getTypeId,0);
}
}); // ==> where name like '巧%' and (parentId > 5 or type_id >= 0)
return this.typeMapper.selectList(lambdaQueryWrapper);
}
- 自定义查询
在Dao层的mapper中自定义查询方法,其中Wrapper参数写法固定
// @Select("select * from user ${ew.customSqlSegment}")
public List<Type> selectAll(@Param(Constants.WRAPPER) Wrapper<Type> wrapper);
Mapper.xml文件中自定义SQL(千万别跟我一样用select * 偷懒)也可以用注解的方式创建SQL语句
<select id="selectAll" resultType="com.example.demo.type.entity.Type">
select * from user ${ew.customSqlSegment}
</select>
Service层代码
public List<Type> selectAllNative(){
QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
queryWrapper.orderByAsc("id");
return this.typeMapper.selectAll(queryWrapper);
}