mybatis使用foreach批次插入,解决sequence只查询一次的问题

oracle的批量插入方式是:
insert  into db(id, zgbh, shbzh)
        select '1', '2', '3' from dual
        union all select '2', '3', '4' from dual
        union all select '3', '4', '5' from dual
        union all select '4', '5', '6' from dual
        union all select '5', '6', '7' from dual

由于项目使用到sequence生成id,刚开始的写法:
<!-- 批次插入,List-->
     <insert id="insertList" useGeneratedKeys="true" parameterType="java.util.List">
        <selectKey resultType="long" keyProperty="id" order="BEFORE">
            SELECT SEQ_xxx_DETAIL.NEXTVAL FROM DUAL
        </selectKey>
         insert into TBL_xxx_DETAIL
              (
                 <include refid="allColumns"/>
              )
              <foreach collection="list" item="item" index="index"  separator="UNION ALL" >
                  SELECT
                 #{id, jdbcType=NUMERIC javaType=long},
                 #{item.batchFundTitleId, jdbcType=NUMERIC javaType=long},
                 #{item.transactionRequestId, jdbcType=NUMERIC javaType=long},
                 #{item.arAmt, jdbcType=DECIMAL javaType=java.math.BigDecimal},
                 #{item.bankServiceAmt, jdbcType=DECIMAL javaType=java.math.BigDecimal},
                 #{item.receivedAmt, jdbcType=DECIMAL javaType=java.math.BigDecimal},
                 #{item.realReceivedAmt, jdbcType=DECIMAL javaType=java.math.BigDecimal},
                 #{item.verifyDate, jdbcType=DATE javaType=date},
                 #{item.verifyOp, jdbcType=VARCHAR javaType=string},
                 #{item.verifyStatus, jdbcType=INTEGER javaType=int},
                 #{item.created, jdbcType=VARCHAR javaType=string},
                 #{item.createdDate, jdbcType=DATE javaType=date},
                 #{item.createdIp, jdbcType=VARCHAR javaType=string},
                 #{item.modified, jdbcType=VARCHAR javaType=string},
                 #{item.modifiedDate, jdbcType=DATE javaType=date},
                 #{item.modifiedIp, jdbcType=VARCHAR javaType=string}
                  from dual
             </foreach>
     </insert>
这样的写法sequence的查询方法只查询一次,造成list中的对象的再插入时id都会一样,违反主键的唯一性约束。

所以修改为如下的形式:
<insert id="insertList" useGeneratedKeys="true" parameterType="java.util.List">
        <selectKey resultType="long" keyProperty="id" order="BEFORE">
            SELECT SEQ_xxx_DETAIL.NEXTVAL FROM DUAL
        </selectKey>

         insert into TBL_xxx_DETAIL
              (
                 <include refid="allColumns"/>
              ) select  SEQ_xxx_DETAIL.NEXTVAL,A.*  from(
              <foreach collection="list" item="item" index="index"  separator="UNION ALL" >
                  SELECT
                 #{item.batchFundTitleId, jdbcType=NUMERIC javaType=long},
                 #{item.transactionRequestId, jdbcType=NUMERIC javaType=long},
                 #{item.arAmt, jdbcType=DECIMAL javaType=java.math.BigDecimal},
                 #{item.bankServiceAmt, jdbcType=DECIMAL javaType=java.math.BigDecimal},
                 #{item.receivedAmt, jdbcType=DECIMAL javaType=java.math.BigDecimal},
                 #{item.realReceivedAmt, jdbcType=DECIMAL javaType=java.math.BigDecimal},
                 #{item.verifyDate, jdbcType=DATE javaType=date},
                 #{item.verifyOp, jdbcType=VARCHAR javaType=string},
                 #{item.verifyStatus, jdbcType=INTEGER javaType=int},
                 #{item.created, jdbcType=VARCHAR javaType=string},
                 #{item.createdDate, jdbcType=DATE javaType=date},
                 #{item.createdIp, jdbcType=VARCHAR javaType=string},
                 #{item.modified, jdbcType=VARCHAR javaType=string},
                 #{item.modifiedDate, jdbcType=DATE javaType=date},
                 #{item.modifiedIp, jdbcType=VARCHAR javaType=string}
                  from dual
             </foreach>) A
     </insert>
把foreach中的id去掉,foreach拼出来的数据作为一张表A,然后从表A中查询数据,再接上从sequence中读取的值作为id。这样sequence的值就会多次读取,id就会不一样。
<selectKey resultType="long" keyProperty="id" order="BEFORE">
            SELECT SEQ_xxx_DETAIL.NEXTVAL FROM DUAL
        </selectKey>
删除后会报错:SQL command not properly ended,暂时没有查出来原因,有知道的原因的请告诉我哈。
以此文抛砖引玉吧,希望能有更合适的方式。

mysql的批量插入如下:
INSERT INTO MyTable(ID,NAME) VALUES(7,'003'),(8,'004'),(9,'005')
而且mysql有自增字段,可以把id设置为自增的,这样的话就不存在id一致的情况。

<insert id="insertBatch" > 
    insert into student ( NAME,SEX,ADDRESS,TELEPHONE,TID)  
    values  
    <foreach collection="list" item="item" index="index" open="(" separator=","  close=")"> 
         #{item.name},
         #{item.sex},
         #{item.address},
         #{item.telephone},
         #{item.tId} 
    </foreach> 
</insert

============================================================================================

create database link DBLinkName  connect to  用户名 identified by 密码
  using '(DESCRIPTION =
    (ADDRESS_LIST =
      (ADDRESS = (PROTOCOL = TCP)(HOST = 192.168.0.1)(PORT = 1521))
    )
    (CONNECT_DATA =
      (SERVICE_NAME = orcl)
    )
  )'

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
### 回答1: Mybatis 中的 `foreach` 元素可以用来循环插入多条数据。 首先,你需要定义一个用于存储数据的列表,并将该列表作为 `foreach` 元素的参数传递进去。比如,如果你要插入的数据是多个用户的信息,你可以这样定义列表: ``` List<User> users; ``` 然后在 `foreach` 元素中使用 `item` 属性来表示列表中的每一个元素,并使用 `index` 属性来表示当前元素的索引。 例如,你可以这样使用 `foreach` 循环插入多条数据: ``` <insert id="insertUsers" parameterType="java.util.List"> insert into users (name, age) values <foreach collection="list" item="item" index="index" separator=","> (#{item.name}, #{item.age}) </foreach> </insert> ``` 在调用该 SQL 语句时,你只需要传入 `users` 列表即可。Mybatis 会自动将列表中的每一个元素插入数据库。 ### 回答2: Mybatis使用foreach循环可以方便地插入多条数据。 在使用Mybatis进行数据库操作时,如果需要插入多条数据,可以使用foreach循环来简化操作。具体步骤如下: 1. 在需要执行插入操作的Mapper接口中,定义一个方法,方法的参数是一个List类型的对象,用于接收需要插入的数据。 2. 在Mapper接口的XML文件中,使用foreach元素来进行循环插入操作,格式如下: ``` <insert id="insertData" parameterType="java.util.List"> INSERT INTO table_name (column1, column2, ...) VALUES <foreach collection="list" item="item" separator=","> (#{item.property1}, #{item.property2}, ...) </foreach> </insert> ``` 在上面的代码中,table_name是需要插入数据的表名,column1, column2等是需要插入的列名,item是循环中的元素,list是传入的参数。 3. 在需要插入数据的代码处,调用Mapper接口中定义的方法,并传入一个包含需要插入的数据的List对象。 通过以上步骤,就可以使用foreach循环来插入多条数据了。在循环中,每次插入的是List中的一个元素,通过#{item.property1}的方式,可以获取到元素中的各个属性值,并插入到对应的列中。 使用Mybatisforeach循环,可以简化多条数据的插入操作,提高开发效率。同时,该方法也适用于更新操作。 ### 回答3: Mybatis使用foreach循环可以方便地循环插入多条数据。在进行批量插入时,可以将一个集合或者数组作为参数传递给Mapper中的方法。通过使用foreach循环,可以遍历集合中的每一个元素,并将其插入到数据库中。 首先,在Mapper的XML配置文件中,可以使用foreach标签来声明循环插入的方式。可以设置collection属性为传入的集合参数的名称,item属性表示循环中每个元素的别名,index属性表示当前元素在集合中的索引。 然后,可以使用foreach标签中的open、close和separator属性来设置插入语句的前缀、后缀和分隔符。 接着,在插入的SQL语句中使用foreach循环的方式,可以通过在values后面添加foreach标签,并在其中通过item属性获取集合中的元素。 最后,在调用Mapper中的方法时,将需要插入的集合作为参数传递给方法即可实现循环插入多条数据。 使用Mybatisforeach循环插入多条数据的好处是可以减少与数据库的交互次数,提高插入数据的效率。通过将多条数据批量插入到数据库中,可以减少插入操作的开销,提高系统的性能。 总之,Mybatis使用foreach循环插入多条数据是一种方便快捷的方式,在插入大量数据时特别有效。同时,在使用foreach循环时,需要注意避免SQL注入问题,保证插入的数据的安全性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值