mybatis错误之in查询 <foreach>循环问题

目录
  • in查询 <foreach>循环问题
    • 1.我就随便用了一种传list,再foreach循环
    • 2.findByCaseNos(Long[] caseNos)
    • 3.findByCaseNos(String name, Long[] caseNos)
  • in查询和foreach标签使用
    • 1.单参数List的类型
    • 2.单参数Array的类型
    • 3.多参数封装成Map的类型
    • 4.嵌套foreach的使用

in查询 <foreach>循环问题

当我在做in查询的时候,发现在网上有很多种写法:

接口

public List<CaseReview > findList(CaseReview caseReview);

public class CaseReview{
private String caseNo;
private List<String> caseNos;//caseNo
//===gettter,setter省略===
}

1.我就随便用了一种传list,再foreach循环

具体如下:

t.case_no in 
<foreach item="item" index="index" collection="caseNos" open="(" separator="," close=")">  
#{item}  
</foreach> 

然而报:

org.apache.ibatis.reflection.ReflectionException: There is no getter for property named '__frch_item_0' in 'class

于是解决办法就出来了:

就是将 #{item} 改为 ‘${item}’就搞定了

<foreach item="item" index="index" collection="caseNos" open="(" separator="," close=")">  
   '${item}'  
</foreach>

如果你非要用#也不是不可以

<foreach item="item" index="index" collection="caseNos" open="(" separator="," close=")">  
             #{caseNos[${index}]}  
</foreach>

还有两种in查询方法,其实都在做foreach循环遍历

2.findByCaseNos(Long[] caseNos)

如果参数的类型是Array,则在使用时,collection属性要必须指定为 array

 caseNos in  
    <foreach item="item" index="index" collection="array" open="(" separator="," close=")">  
     #{caseNos}
    </foreach>  

3.findByCaseNos(String name, Long[] caseNos)

当查询的参数有多个时:

这种情况需要特别注意,在传参数时,一定要改用Map方式, 这样在collection属性可以指定名称

Map<String, Object> params = new HashMap<String, Object>();
 params.put("name", name);
 params.put("ids", caseNos);
 mapper.findByIdsMap(params);
 caseNos in  
     <foreach item="item" index="index" collection="caseNos" open="(" separator="," close=")">  
      #{item}  
     </foreach> 

in查询和foreach标签使用

Mybatis中的foreach的主要用在构建in条件中,它可以在SQL语句中进行迭代一个集合。

foreach元素的属性主要有 item,index,collection,open,separator,close:

  • item:表示集合中每一个元素进行迭代时的别;
  • index:指定一个名字,用于表示在迭代过程中,每次迭代到的位置;
  • open:表示该语句以什么开始;
  • separator:表示在每次进行迭代之间以什么符号作为分隔 符;
  • close:表示以什么结束;
  • collection:在使用foreach的时候最关键的也是最容易出错的就是collection属性,该属性是必须指定的,但是在不同情况 下,该属性的值是不一样的,主要有一下3种情况:

1. 如果传入的是单参数且参数类型是一个List的时候,collection属性值为list

2. 如果传入的是单参数且参数类型是一个array数组的时候,collection的属性值为array

3.  如果传入的参数是多个的时候,我们就需要把它们封装成一个Map了,当然单参数也可以封装成map,实际上如果你在传入参数的时候,在breast里面也是会把它封装成一个Map的,map的key就是参数名,所以这个时候collection属性值就是传入的List或array对象在自己封装的map里面的key

下面分别来看看上述三种情况的示例代码:

1.单参数List的类型

             <select id="dynamicForeachTest" resultType="Blog">
                    select * from t_blog where id in
                              <foreach collection="list" index="index" item="item" open="(" separator="," close=")">
                                        #{item}
                              </foreach>
           </select>

上述collection的值为list,对应的Mapper是这样的:

public List<Blog> dynamicForeachTest(List<Integer> ids);

测试代码:

              @Test
              public void dynamicForeachTest() {
                       SqlSession session = Util.getSqlSessionFactory().openSession();
                       BlogMapper blogMapper = session.getMapper(BlogMapper.class);
                       List<Integer> ids = new ArrayList<Integer>();
                       ids.add(1);
                       ids.add(3);
                       ids.add(6);
                       List<Blog> blogs = blogMapper.dynamicForeachTest(ids);
                       for (Blog blog : blogs){
                              System.out.println(blog);
                       }
                        session.close();
             }

2.单参数Array的类型

             <select id="dynamicForeach2Test" resultType="Blog">
                     select * from t_blog where id in
                             <foreach collection="array" index="index" item="item" open="(" separator="," close=")">
                                    #{item}
                            </foreach>
            </select>

上述collection为array,对应的Mapper代码:

public List<Blog> dynamicForeach2Test(int[] ids);

对应的测试代码:

@Test
         public void dynamicForeach2Test() {
                  SqlSession session = Util.getSqlSessionFactory().openSession();
                  BlogMapper blogMapper = session.getMapper(BlogMapper.class);
                  int[] ids = new int[] {1,3,6,9};
                  List<Blog> blogs = blogMapper.dynamicForeach2Test(ids);
                  for (Blog blog : blogs){
                          System.out.println(blog);
                  }
                 session.close();
           }

3.多参数封装成Map的类型

         <select id="dynamicForeach3Test" resultType="Blog">
                    select * from t_blog where title like "%"#{title}"%" and id in
                             <foreach collection="ids" index="index" item="item" open="(" separator="," close=")">
                                       #{item}
                            </foreach>
        </select>

上述collection的值为ids,是传入的参数Map的key,对应的Mapper代码:

public List<Blog> dynamicForeach3Test(Map<String, Object> params);

对应测试代码:

@Test
        public void dynamicForeach3Test() {
                 SqlSession session = Util.getSqlSessionFactory().openSession();
                 BlogMapper blogMapper = session.getMapper(BlogMapper.class);
                 final List<Integer> ids = new ArrayList<Integer>();
                 ids.add(1);
                 ids.add(2);
                 ids.add(3);
                 ids.add(6);
                 Map<String, Object> params = new HashMap<String, Object>();
                 params.put("ids", ids);
                 params.put("title", "中国");
                 List<Blog> blogs = blogMapper.dynamicForeach3Test(params);
                 for (Blog blog : blogs)
                       System.out.println(blog);
                 session.close();
        }

4.嵌套foreach的使用

map 数据如下 Map<String,List<Long>>

测试代码如下:

       public void getByMap(){
        Map<String,List<Long>> params=new HashMap<String, List<Long>>();
        List<Long> orgList=new ArrayList<Long>();
        orgList.add(10000003840076L);
        orgList.add(10000003840080L);
        
        List<Long> roleList=new ArrayList<Long>();
        roleList.add(10000000050086L);
        roleList.add(10000012180016L);
        
        params.put("org", orgList);
        params.put("role", roleList);
        
        List<BpmDefUser> list= bpmDefUserDao.getByMap(params);
        System.out.println(list.size());
    }

dao代码如下:

public List<BpmDefUser> getByMap(Map<String,List<Long>> map){
            Map<String,Object> params=new HashMap<String, Object>();
            params.put("relationMap", map);
            return this.getBySqlKey("getByMap", params);
    }

xml代码如下:

<select id="getByMap" resultMap="BpmDefUser">
        
            <foreach collection="relationMap" index="key"  item="ent" separator="union">
                SELECT *
                FROM BPM_DEF_USER
                where  RIGHT_TYPE=#{key}
                and OWNER_ID in 
                <foreach collection="ent"  item="id" separator="," open="(" close=")">
                    #{id}
                </foreach>
            </foreach>
        
    </select>

index 作为map 的key。item为map的值,这里使用了嵌套循环,嵌套循环使用ent。

《项目实践》

@Override
public Container<Map<String,Object>> findAuditListInPage(
  Map<String, Object> params) {
  //1、参数组装
  PageModel pageMode = new PageModel();
  try {
    if(params.get("page")!=null){
      pageMode.setPage(Integer.parseInt(params.get("page").toString()));
    }
    if(params.get("rows")!=null){
      pageMode.setRows(Integer.parseInt(params.get("rows").toString()));
    }
  } catch (Exception e) {
    Assert.customException(RestApiError.COMMON_ARGUMENT_NOTVALID);
  }
  //分页条件组装
  pageMode.putParam(params);
  if(params.get("startCreateTime") !=null){
    Date parse = DateUtil.parse(params.get("startCreateTime").toString(), DateUtil.yyyyMMddHHmmss);
    params.put("startCreateTime",parse);
  }
  if(params.get("endCreateTime") !=null){
    Date parse = DateUtil.parse(params.get("endCreateTime").toString(), DateUtil.yyyyMMddHHmmss);
    params.put("endCreateTime",parse);
  }
  if(params.get("type") !=null){              //type可以多选
    String typeString = params.get("type").toString();
    String typeArray [] = typeString.split(",");
    params.put("type", typeArray);
  }
  if(params.get("state") !=null){             //state可以多选
     String stateString = params.get("state").toString();
     if(stateString.equals(DictConstants.APPLICATION_STATE.AUDITING)
               ||stateString.equals(DictConstants.APPLICATION_STATE.WAITING_AUDIT)){
      stateString = "waitingAudit,auditing";
  }
  String stateArray [] = stateString.split(",");
  params.put("state", stateArray);
  }
  //分页数据组装
  Container<Map<String,Object>> container = new Container<Map<String,Object>>();
  List<Map<String,Object>> auditModelList = cmApplicationRepo.findAuditList(params);
  for(Map<String,Object> audit:auditModelList){
    //设置是否关注过
    Long auditId = Long.parseLong(audit.get("auditId").toString());
    Long auditPersonId = Long.parseLong(params.get("auditPersonId").toString());
    Map<String, Object> followMap = new HashMap<String,Object>();
    followMap.put("sourceType", DictConstants.FOLLOW_SOURCE_TYPE.FOLLOW_APPLICATION);
    followMap.put("sourceId", auditId);
    followMap.put("userId", auditPersonId);
    List<BizFollowModel> followList = bizFollowService.find(followMap);
    if(followList!= null && followList.size()>0){
      audit.put("isFollow", "true");
    }else{
      audit.put("isFollow", "false");
    }
  }
  container.setList(auditModelList);
  container.setTotalNum(cmApplicationRepo.countAuditListNumber(params));
  return container;
}

DAO

@Override
public List<Map<String,Object>> findAuditList(Map<String, Object> map) {
  return findList("getAuditList", map);
}

xml

<!-- 查询申请列表-->
<select id="getApplicationList" resultType="java.util.Map" parameterType="map">
  select
    a.ID AS id,
    a.STATE AS stateCode, b.DICT_VALUE AS stateValue,
    a.ITEM AS itemCode, c.DICT_VALUE AS itemValue,
    a.TYPE AS typeCode, d.DICT_VALUE AS typeValue,
    a.APP_PERSON_ID AS appPersonId,
    a.CREATE_TIME AS createTime
  from cm_application a
  LEFT JOIN cm_dict_type b on a.STATE = b.DICT_CODE AND b.TYPE = 'Application_State'
  LEFT JOIN cm_dict_type c on a.ITEM = c.DICT_CODE
  LEFT JOIN cm_dict_type d on a.TYPE = d.DICT_CODE
  where 1=1
  <if test="item != null" >
    and a.ITEM = #{item,jdbcType=VARCHAR}
    </if>
    <if test="type != null" > 
      and a.TYPE IN 
      <foreach item="typeArray" index="index" collection="type" open="(" separator="," close=")">
        #{typeArray}
      </foreach>
    </if>
    <if test="appPersonId != null" >
      and a.APP_PERSON_ID = #{appPersonId,jdbcType=BIGINT}
    </if>
    <if test="state != null" >
      and a.STATE IN
      <foreach item="stateArray" index="index" collection="state" open="(" separator="," close=")">
        #{stateArray}
      </foreach>
    </if>
    <!-- 分页查询时,要选择createTime在starCreateTime和endCreatetTime之间的记录 -->
    <if test="startCreateTime != null" >
      and a.CREATE_TIME &gt;= #{startCreateTime,jdbcType=TIMESTAMP}
    </if>
    <if test="endCreateTime != null" >
      and a.CREATE_TIME &lt;= #{endCreateTime,jdbcType=TIMESTAMP}
    </if>
  order by a.ID
  <include refid="Paging" />
</select>

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • MyBatis foreach 批量更新实例

    在做配置选项(设备类型,所属楼层等)的时候,当删除某配置的时候,我需要检验该配置是否已被删除. @Override public BaseVO deleteOptionDetail(Integer id) { // 合法性验证 if (null == id) { return ParamErrorVO.getInstance(); } ConfigOptionDetail configOptionDetail = configOptionDetailMapper.selectById(id);

  • MyBatis中使用foreach循环的坑及解决

    目录 使用foreach循环的坑 正确写法为 忽现的Mybatis foreach失效记录 解决方案 使用foreach循环的坑 我们首先看一段MyBatis中使用foreach循环的sql: SELECT * FROM table where id in <foreach item="item" collection="ids1" open="(" close=")" index="0" separ

  • mybatis 遍历foreach中or拼接的操作

    我就废话不多说了,大家还是直接看看关键代码吧: select id, name from t_keys where 1 = 1 <if test="keys != null"> <foreach collection="keys" item="key" open="AND (" close=")" separator="or" > name = #{key} &l

  • mybatis foreach 属性及其三种使用情况详解

    目录 foreach 属性介绍 单参数是 array 类型 单参数是 List 类型 单参数是 Map 类型 多参数 foreach 属性介绍 foreach 用于迭代传入过来的参数. 它的属性介绍分别是 collection:表示传入过来的参数的数据类型.该参数为必选.要做 foreach 的对象,作为入参时,List 对象默认用 list 代替作为键,数组对象有 array 代替作为键,Map 对象没有默认的键.当然在作为入参时可以使用 @Param(“keyName”) 来设置键,设置 k

  • 基于mybatis注解动态sql中foreach工具的方法

    目录 实现目标 工具类 测试 集成进spring后的使用方法 Mapper层 Provider层 Ognl问题 解决方案 创建一个DefaultMemberAccess.java文件 改造ForeachMybatisUtils.java类 实现目标 由于在注解版mybatis中手动循环拼接动态sql容易出错 请看mybatis注解动态sql注入map和list(防sql注入攻击),所以封装了这个类似于foreach标签的工具方法. 由于mybatis(3.5.6.3.5.7 or earlier

  • 深入浅析MyBatis foreach标签

    前面我们学习了如何使用 Mybatis if.where.trim 等动态语句来处理一些简单的查询操作.对于一些 SQL 语句中含有 in 条件,需要迭代条件集合来生成的情况,可以使用 foreach 来实现 SQL 条件的迭代. Mybatis foreach 标签用于循环语句,它很好的支持了数据和 List.set 接口的集合,并对此提供遍历的功能.语法格式如下. SELECT * FROM product_ WHERE ID in <foreach item="item"

  • mybatis错误之in查询 <foreach>循环问题

    目录 in查询 <foreach>循环问题 1.我就随便用了一种传list,再foreach循环 2.findByCaseNos(Long[] caseNos) 3.findByCaseNos(String name, Long[] caseNos) in查询和foreach标签使用 1.单参数List的类型 2.单参数Array的类型 3.多参数封装成Map的类型 4.嵌套foreach的使用 in查询 <foreach>循环问题 当我在做in查询的时候,发现在网上有很多种写法:

  • MyBatis实现批量插入数据,多重forEach循环

    目录 批量插入数据,多重forEach循环 下面是一个实际应用 mybatis insert foreach 项目场景 批量插入数据,多重forEach循环 在业务开发过程中,遇到批量插入时,需要进行多重forEach循环的情况. 下面是一个实际应用 public class SysRoleData extends DataEntity<SysRoleData> {    private static final long serialVersionUID = 1L;    private S

  • MyBatis如何进行双重foreach循环

    目录 进行双重foreach循环 使用场景 mybatis foreach循环,传入多个参数 上代码 进行双重foreach循环 mapname是一个Map<String,Map<String,Object>> 对象 <foreach collection="mapname" index="key1" item="value1" separator=",">     <foreach

  • mybatis foreach 循环 list(map)实例

    目录 foreach 循环 list(map) 整体需求就是 mybatis.foreach循环如下 使用foreach处理list中的map Mapper中的方法是 Mapper.xml的sql是 foreach 循环 list(map) 直接上代码: 整体需求就是 1.分页对象里面有map map里面又有数组对象 2.分页对象里面有list list里面有map map里面有数组对象. public class Page { private Map maps; private List li

  • mybatis insert foreach循环插入方式

    mybatis insert foreach循环插入 @Insert("<script>" + "insert into driver_account_appeal_photo (appeal_id,appeal_photo_path) values\n" + "<foreach collection=\"photoList\" item=\"item\" index=\"index\&q

  • mybatis多个区间处理方式(双foreach循环)

    目录 mybatis多个区间处理 思路 多个foreach同时使用问题 应用场景 注意事项 解决方案 mybatis多个区间处理 如图:要实现车辆数不同区间查询条件 思路 a.前端传数组,数组里面放"1-5"String类型值             b.后端mybatis用双foreach循环解析     后端代码如下: <!--图例车辆数区间--> <if test="countCargoList != null and countCargoList.s

  • 详解JAVA中的for-each循环与迭代

    在学习java中的collection时注意到,collection层次的根接口Collection实现了Iterable<T>接口(位于java.lang包中),实现这个接口允许对象成为 "foreach" 语句的目标,而此接口中的唯一方法,实现的就是返回一个在一组 T 类型的元素上进行迭代的迭代器. 一.迭代器Iterator 接口:Iterator<T> public interface Iterator<E>{ boolean hasNext

  • Java for-each循环使用难题2例(高级使用方法)

    Java中,for-each循环简化了任何Collection或array的遍历过程,但并不是每个Java程序员都了解本文将要描述的for-each 循环的一些细节.与 Java5 发布的其他术语:释放别名泛型,自动封装和可变参数不同,Java开发者对for-each循环的使用比任何其他特性更加频繁,但当问及高级的for-each循环怎样工作,或什么是在for-each循环中使用Collection时的基本需求时,就不是每个人都能够回答的了. 本篇教程和例子旨在通过深入研究for-each 循环

  • js的for in循环和java里foreach循环的区别分析

    本文实例分析了js的for in循环和java里foreach循环的区别.分享给大家供大家参考.具体分析如下: js里的for in循环定义如下: 复制代码 代码如下: for(var variable in obj) { ... } obj可以是一个普通的js对象或者一个数组.如果obj是js对象,那么variable在遍历中得到的是对象的属性的名字,而不是属性对应的值.如果obj是数组,那么variable在遍历中得到的是数组的下标. 遍历对象实验: 复制代码 代码如下: var v = {

随机推荐