作者:学无先后 达者为先

作者:偶尔记一下

foreach一共有三种类型,分别为List,[](array),Map三种。

下面表格是我总结的各个属性的用途和注意点。

foreach属性

属性 描述
item 循环体中的具体对象。支持属性的点路径访问,如item.age,item.info.details。
具体说明:在list和数组中是其中的对象,在map中是value。
该参数为必选。
collection 要做foreach的对象,作为入参时,List<?>对象默认用list代替作为键,数组对象有array代替作为键,Map对象没有默认的键。
当然在作为入参时可以使用@Param("keyName")来设置键,设置keyName后,list,array将会失效。 除了入参这种情况外,还有一种作为参数对象的某个字段的时候。举个例子:
如果User有属性List ids。入参是User对象,那么这个collection = "ids"
如果User有属性Ids ids;其中Ids是个对象,Ids有个属性List id;入参是User对象,那么collection = "ids.id"
上面只是举例,具体collection等于什么,就看你想对那个元素做循环。
该参数为必选。
separator 元素之间的分隔符,例如在in()的时候,separator=","会自动在元素中间用“,“隔开,避免手动输入逗号导致sql错误,如in(1,2,)这样。该参数可选。
open foreach代码的开始符号,一般是(和close=")"合用。常用在in(),values()时。该参数可选。
close foreach代码的关闭符号,一般是)和open="("合用。常用在in(),values()时。该参数可选。
index 在list和数组中,index是元素的序号,在map中,index是元素的key,该参数可选。

注:Map类型没有默认的map,所以不能直接写collection="map",如果这么写,需要保证传入的Map参数有@Param("map")注解。

有关参数的更详细内容,建议看:深入了解MyBatis参数

这节讲的是foreach中map的用法。

map和List,array相比,map是用K,V存储的,在foreach中,使用map时,index属性值为map中的Key的值。

因为map中的Key不同于list,array中的索引,所以会有更丰富的用法。

第一个简单例子:

<insert id="ins_string_string">
insert into string_string (key, value) values
<foreach item="item" index="key" collection="map"
open="" separator="," close="">(#{key}, #{item})</foreach>
</insert>

可以看到这个例子相当简单,表中需要两个值,正好和K,V对应,因而map中的一个K,V就对应一条数据,如果map中有多个K,V,就会保存多个结果。

如果map中有两对K,V,那么执行SQL如下:

         [plain] view plain copy
  1. DEBUG [main] - ==>  Preparing: insert into string_string (key, value) values (?, ?) , (?, ?)
  2. DEBUG [main] - ==> Parameters: key 1(String), value 1(String), key 2(String), value 2(String)
  3. DEBUG [main] - <==    Updates: 2

大部分数据库是支持values()()这种形式的插入语句,可以插入多条(相关链接 - 可能需翻墙)。

下面再看一个select的例子:

  1. <select id="sel_key_cols" resultType="int">
  2. select count(*) from key_cols where
  3. <foreach item="item" index="key" collection="map"
  4. open="" separator="AND" close="">${key} = #{item}</foreach>
  5. </select>

可以看到这里用key=value来作为查询条件,对于动态的查询,这种处理方式可以借鉴。一定要注意到$和#的区别,$的参数直接输出,#的参数会被替换为?,然后传入参数值执行。

上述SQL执行日志如下:

  1. DEBUG [main] - ==>  Preparing: select count(*) from key_cols where col_a = ? AND col_b = ?
  2. DEBUG [main] - ==> Parameters: 22(Integer), 222(Integer)
  3. DEBUG [main] - <==      Total: 1

最后,如果不考虑元素的顺序和map中Key,map和list,array可以拥有一样的效果,都是存储了多个值,然后循环读取出来。

foreach一共有三种类型,分别为List,[](array),Map三种。

 示例一:

<select id="countByUserList" resultType="_int" parameterType="list">
select count(*) from users
<where>
id in
<foreach item="item" collection="list" separator="," open="(" close=")" index="">
#{item.id, jdbcType=NUMERIC}
</foreach>
</where>
</select>

注:select count(*) from users WHERE id in ( ? , ? )

示例二:

 <select id="selectStorageProductInventroy" parameterType="java.util.Map" resultType="StorageProductInventroyModel">
select op.inventroy_operation_id ,
op.inventroy_id ,
ty.name as operation_type,
DATE_FORMAT(op.insert_time,'%Y/%m/%d') as insert_time,
em.name as employee,
sm1.name as from_store,
st1.name as to_store,
sm.name as from_storage,
st.name as to_storage,
hi.number,
pr.name as product_name,
pr.category_id as categoryString,
pr.product_code,
hi.amount,
hi.back_amount,
hi.unit,
re.description as operation_reason,
hi.description from inventroy_history hi left join inventroy_operation op
on hi.inventroy_operation_id = op.inventroy_operation_id left join product pr
on hi.product_id =pr.product_id left join operation_type ty
on op.operation_type_id = ty.operation_type_id left join employee em
on op.employee_id = em.employee_id left join (select inventroy_operation.to_storage_id as to_storage_id,storage.name as name
from inventroy_operation , storage
where inventroy_operation.to_storage_id = storage.storage_id) st
on op.to_storage_id = st.to_storage_id left join (select inventroy_operation.from_storage_id as from_storage_id,storage.name as name
from inventroy_operation , storage
where inventroy_operation.from_storage_id = storage.storage_id) sm
on op.from_storage_id = sm.from_storage_id left join (select inventroy_operation.to_store_id as to_store_id,store.name as name
from inventroy_operation , store
where inventroy_operation.to_store_id = store.store_id) st1
on op.to_store_id = st1.to_store_id left join (select inventroy_operation.from_store_id as from_store_id,store.name as name
from inventroy_operation , store
where inventroy_operation.from_store_id = store.store_id) sm1
on op.from_store_id = sm1.from_store_id left join operation_reason re
on hi.operation_reason_id = re.operation_reason_id
WHERE 1=1
<if test="operation_type_id != null and operation_type_id.length != 0">
and
<foreach collection="operation_type_id" item="operationTypeId" index="index" open="(" separator="or" close=")">
op.operation_type_id = #{operationTypeId}
</foreach>
</if>
<if test="storage_id != 0 and storage_id != null">
and (op.from_storage_id = #{storage_id} or op.to_storage_id = #{storage_id})
</if>
<if test="category_id != 0 and category_id != null">
and pr.category_id like concat('%','(',#{category_id},')','%')
</if>
<if test="product_code != null">
and pr.product_code like concat('%',#{product_code},'%')
</if>
<if test="product_name != null">
and pr.name like concat('%',#{product_name},'%')
</if>
and date(op.insert_time) between #{start_time} and #{end_time}
group by hi.inventroy_history_id
</select>

注:and (op.operation_type_id = ? or op.operation_type_id = ?)

示例三:数组

 public void testQuery() {
ColInfoDao dao=(ColInfoDao)ctx.getBean("colInfoDao");
Map map = new HashMap();
map.put("userId", "tom");
map.put("password", "123");
String[] a = { "20000001", "20000002" };
map.put("classIds", Arrays.asList(a));
Object password = dao.query(map);
System.out.println("password:" + password);
Assert.assertEquals("123", password);
}

XML:(感觉不适合mybatis,可以使用在ibatis中,iBatis 2.x 和 MyBatis 3.0.x)

<select id="queryPasswordByUserId" parameterClass="java.util.Map"      resultClass="java.lang.String">
<![CDATA[
select PASSWORD as password from T_S_P_USER
]]>
<dynamic prepend="where">
<isNotEmpty prepend="AND" property="userId">
USER_ID=#userId#
</isNotEmpty>
<isNotEmpty prepend="AND" property="password">
PASSWORD=#password#
</isNotEmpty>
<isNotEmpty prepend="AND" property="classIds">
<iterate property="classIds" open="(" conjunction="OR" close=")">
CLASS_ID = #classIds[]#
</iterate>
</isNotEmpty>
</dynamic>
</select>

示例四:Map

map和List,array相比,map是用K,V存储的,在foreach中,使用map时,index属性值为map中的Key的值。

因为map中的Key不同于list,array中的索引,所以会有更丰富的用法。

<insert id="ins_string_string">
insert into string_string (key, value) values
<foreach item="item" index="key" collection="map"
open="" separator="," close="">(#{key}, #{item})</foreach>
</insert>

可以看到这个例子相当简单,表中需要两个值,正好和K,V对应,因而map中的一个K,V就对应一条数据,如果map中有多个K,V,就会保存多个结果。

如果map中有两对K,V,那么执行SQL如下:

DEBUG [main] - ==>  Preparing: insert into string_string (key, value) values (?, ?) , (?, ?)
DEBUG [main] - ==> Parameters: key 1(String), value 1(String), key 2(String), value 2(String)
DEBUG [main] - <== Updates: 2

下面再看一个select的例子:

<select id="sel_key_cols" resultType="int">
select count(*) from key_cols where
<foreach item="item" index="key" collection="map"
open="" separator="AND" close="">${key} = #{item}</foreach>
</select>

可以看到这里用key=value来作为查询条件,对于动态的查询,这种处理方式可以借鉴。一定要注意到$和#的区别,$的参数直接输出,#的参数会被替换为?,然后传入参数值执行。

DEBUG [main] - ==>  Preparing: select count(*) from key_cols where col_a = ? AND col_b = ?
DEBUG [main] - ==> Parameters: 22(Integer), 222(Integer)
DEBUG [main] - <== Total: 1

最后,如果不考虑元素的顺序和map中Key,map和list,array可以拥有一样的效果,都是存储了多个值,然后循环读取出来。

mybatis中foreach使用方法的更多相关文章

  1. mybatis中foreach的用法(转)

    foreach一共有三种类型,分别为List,[](array),Map三种. foreach属性 属性 描述 item 循环体中的具体对象.支持属性的点路径访问,如item.age,item.inf ...

  2. mybatis中的updateByExampleSelective方法怎么使用

    mybatis中的updateByExampleSelective方法怎么使用.sendDetailMapper.updateByExampleSelective(sendDetail, m);参数m ...

  3. MyBatis中foreach循环的用法

    一.在了解foreach之前,先了解一下mybatis传入参数及parameterType 1.我们在Dao层向对应的mapper.xml文件传递参数时,可以传递的参数有: ①.基本数据类型(如int ...

  4. mybatis中foreach使用

    mybatis中的<foreach collection="list" item="item" index="index" open= ...

  5. 谈一下思考,关于mybatis中<foreach collection="list">中list得来的原因 没看到官方说明

    <foreach> 是在sql语句中进行多个id查询 时用到的,因为mybatis代替jdbc和hibernate, 使用 在xml文件中编写sql语句,这是一个标签文件.然后在 dao层 ...

  6. mybatis 中 foreach 的性能问题及调优

    1.mybatis中最初的sql语句 SELECT 参数1, 参数2, 参数3 FROM 表 WHERE 条件参数1 in <foreach item="item" inde ...

  7. mybatis中foreach参数过多效率很慢的优化

    foreach 后面in 传入的参数有1万条,#和$是有效率区别的,$的效率远高于#,上篇文章做了比较. 但没达到我的理想结果. 1. 更改方式,把foreach 去掉,改成拼装方式, 参数直接拼装成 ...

  8. 用mybatis中的insert方法插入数据,返回值为1,但数据库却没有数据

    刚才在写东西的时候,用mybatis中的 <insert id="add" parameterType="cn.entity.Computer"> ...

  9. mybatis 中 foreach collection的三种用法(转)

    文章转自 https://blog.csdn.net/qq_24084925/article/details/53790287 oreach的主要用在构建in条件中,它可以在SQL语句中进行迭代一个集 ...

随机推荐

  1. java动态更新枚举类

    工作中遇到需要对枚举类的值进行动态更新 手动改不现实也不方便 现记录下来方便以后学习使用 1.在工程utils包中添加动态更新枚举类得工具类(根据自己得项目,放到指定位置调用就可以) 2.一开始陷入了 ...

  2. (十二)mybatis 查询缓存

    目录 什么是查询缓存 图解查询缓存 一级缓存 二级缓存 禁用二级缓存 刷新缓存 二级缓存应用场景 二级缓存局限性 什么是查询缓存 mybatis 在查询数据的时候,会将数据存储起来,下次再次查询相同的 ...

  3. PAT题目AC汇总(待补全)

    题目AC汇总 甲级AC PAT A1001 A+B Format (20 分) PAT A1002 A+B for Polynomials(25) PAT A1005 Spell It Right ( ...

  4. php常用扩展有哪些

    bcmath(精确数值处理) bz2 calendar Core ctype curl date dom ereg exif fileinfo filter ftp gettext hash icon ...

  5. Go-函数高级使用-条件分支-包管理-for循环-switch语句-数组及切片-与或非逻辑符

    目录 科普 python 注释 # 函数高级 if else if else 包管理 下载第三方包 比较热门的框架 for 循环 for 循环的几种写法 switch 语句 数组及数组切片 数组迭代 ...

  6. vc++6.0中查看函数栈的结构

    栈:一种后进先出的数据结构   比如:弹夹 函数调用的约定 传参顺序 传参媒介 如何传递返回值 平衡参数(堆栈平衡):有且只有被调方(callee)和调用方(caller)一方执行 _cdell (c ...

  7. java 内部类、匿名内部类

    一:内部类 1:什么是内部类? 大部分时候,类被定义成一个独立的程序单元.在某些情况下,也会把一个类放在另一个类的内部定义,这个定义在其他类内部的类就被称为内部类(有些地方也叫做嵌套类),包含内部类的 ...

  8. Mancala II

    题目描述 Mancala is a family of board games played around the world, sometimes called sowing games, or c ...

  9. idea插件,配置

    1.插件 2.配置 1.去除double shift 全局搜索的功能,反正不符合我的习惯 Help -> Find Action  搜索 registry ,勾选 double.click.ha ...

  10. MongoDB实战读书笔记(一):JavaScript shell操作MongoDB

    1 基本增删改查 基本概念: 数据库:同关系型数据库 集合:类似关系型数据库的表 文档:类似关系型数据库的行 字段:类似关系型数据库的列 操作: insert:新增,若新增数据的主键已经存在,则会抛异 ...