前言

上节为通过id、idList、idSet、Map设置条件进行等值查询,但在实际使用中,还需要使用> <等连接符进行查询,那么此次则介绍条件构造器这一使用方法。
这个是自己学习时候记得笔记要是想详细了解可以去MP官网,上边有更详细的配置流程以及视频教学:MyBatis-Plus

条件构造器使用(QueryWrapper)

  1. 执行单表操作的 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);

    }
  1. 查询条件中含有 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);
    }
  1. 同时含有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);
    }
  1. 同时含有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);
    }
  1. 同时含有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);
    }
  1. 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);
    }
  1. 只需要返回结果中满足条件的一条 (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);
    }
  1. 指定返回字段
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);
    }
  1. 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);
    }
  1. 构造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);
    }
  1. 条件构造器: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);
    }
  1. 其他查询方法: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);
    }
  1. 其他查询方法: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);
    }
  1. 其他查询方法:selectCount
public Integer selectWrapperCount(){
        QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
        queryWrapper.like("name","巧克力")
                .lt("parentId",20);
        return this.typeMapper.selectCount(queryWrapper);
    }
  1. 其他查询方法:selectOne
    只查询/返回一条数据,多了会报错
    预期返回一条数据或者null
public Type selectWrapperOne(){
        QueryWrapper<Type> queryWrapper = new QueryWrapper<>();
        queryWrapper.like("name","巧克力")
                .lt("parentId",20);
        return this.typeMapper.selectOne(queryWrapper);
    }
  1. 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);
    }
  1. 自定义查询
    在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);
    }