数据库--MyBatis的(insert,update,delete)三种批量操作

转自:http://blog.csdn.net/starywx/article/details/23268465

前段时间由于项目赶期没顾上开发过程中的性能问题,现对部分代码进行优化的过程中发现在数据量大的情况下对数据的操作反应似乎有些慢,就想到对数据库DML操作的时候进行批量操作。说道这里也想到自己在一次面试的时候别问道过批量操作数据的问题。
现对运用说明记录如下:
批量插入insert
方法一:
<insert >
  <selectKey keyProperty="id" order="AFTER"
  resultType="int">
  SELECT LAST_INSERT_ID()
  </selectKey>
   INSERT INTO  sourcedoc (
sdate, sweek, 
      roomno, daysched, nightsched, 
      adminsched, vacationsched, programdept, 
      programname 
)values
<foreach collection="list" item="item" index="index" separator=",">
(
#{item.sdate},#{item.sweek},#{item.roomno},
#{item.daysched},#{item.nightsched},#{item.adminsched},
#{item.vacationsched},#{item.programdept},#{item.programname}
        )
</foreach>
</insert>

方法二: <insert > insert into table1(sdate,sweek,roomno,daysched,nightsched,adminsched,vacationsched,programdept,programname) <foreach collection="list" item="item" index="index" separator="union all"> select #{item.sdate,jdbcType=VARCHAR}, #{item.sweek,jdbcType=VARCHAR}, #{item.roomno,jdbcType=VARCHAR}, #{item.nightsched,jdbcType=VARCHAR}, #{item.adminsched,jdbcType=VARCHAR}, #{item.vacationsched,jdbcType=VARCHAR}, #{item.programdept,jdbcType=VARCHAR},0,0, #{item.programname,jdbcType=VARCHAR} from dual </foreach> </insert> 可以考虑用union all来实现批量插入。 例如: insert into XX_TABLE(XX,XX,XX)select 'xx','xx','xx' union all select 'xx','xx','xx' union all select 'xx','xx','xx' ... 先拼装好语句再动态传入insert into XX_TABLE(XX,XX,XX)后面部分
批量删除(delete) <!-- 通过主键集合批量删除记录 --> <delete > DELETE FROM LD_USER WHERE ID in <foreach item="item" index="index" collection="list" open="(" separator="," close=")"> #{item} </foreach> </delete>
批量修改(update) <update > update orders set state = '0' where no in <foreach collection="list" item="nos" open="(" separator="," close=")"> #{nos} </foreach> </update>
MyBatis中in子句 in 参数 使用方法 1.只有一个参数 参数的类型要声明为List或Array Sql配置如下: <select > SELECT * FROM PRODUCT WHERE PRODUCTNO IN <foreach item="productNo" index="index" collection="参数的类型List或array"> #{productNo} </foreach> </select>
2.多个参数 首先要将多个参数写入同一个map,将map作为一个参数传入mapper Sql配置如下: <select > SELECT * FROM PRODUCT WHERE PRODUCTNO IN <foreach item="id" index="index" collection="map中集合参数的名称"> #{id} </foreach> </select> 批量数据操作的体会 MyBatis的前身就是著名的Ibatis,不知何故脱离了Apache改名为MyBatis。 MyBatis所说是轻量级的ORM框架,在网上看过一个测试报告,感觉相比于Hibernate来说,优势并不明显。 下面说一下比较有趣的现象,根据MyBatis的官方文档,在获得sqlSession时,它有为批量更新而专门准备的: session = sessionFactory.openSession();//用于普通update session = sessionFactory.openSession(ExecutorType.BATCH, true);//用于批量update 一般来说,对MYSQL数据库批量操作时速度取决于,是为每一个处理分别建立一个连接,还是为这一批处理一共建立一个连接。按MyBatis的手册说明,选择ExecutorType.BATCH意味着,获得的sqlSession会批量执行所有更新语句。不过我测试了一下,批量插入1000条数据,发觉ExecutorType.BATCH方式的效率居然比普通的方式差很多。我测试用的Mapper中的insert配置如下,再用for循环插入1000条记录: <insert > insert into student (id, name, sex, address, telephone, t_id ) values (#{id,jdbcType=INTEGER}, #{name,jdbcType=VARCHAR}, #{sex,jdbcType=VARCHAR}, #{address,jdbcType=VARCHAR}, #{telephone,jdbcType=VARCHAR}, #{tId,jdbcType=INTEGER} ) </insert> 关于数据库批量插入时sql语句级的优化,我特意测试了两种方式,在StudentMapper中配置了两种insert模式。第一种对应insert value1,insert value2,,,,;第二种对应insert values (value1, value2,....)。发现后者果然比前者快很多啊。下面是两种insert模式,及测试结果对应图: <!-- 批量 ,传入一个长度为1000的list --> insert into student ( <include ref/> ) values <foreach collection="list" item="item" index="index" separator=","> (null,#{item.name},#{item.sex},#{item.address},#{item.telephone},#{item.tId}) </foreach> </insert> <!-- 在外部for循环调用一千次 --> <insert > insert into student (id, name, sex, address, telephone, t_id ) values (#{id,jdbcType=INTEGER}, #{name,jdbcType=VARCHAR}, #{sex,jdbcType=VARCHAR}, #{address,jdbcType=VARCHAR}, #{telephone,jdbcType=VARCHAR}, #{tId,jdbcType=INTEGER} ) </insert>