Mybatis结合Oracle的foreach insert批量插入

标签: Mybatis批量插入oracle

2016-08-25 21:47 15995人阅读 评论(8) 收藏 举报

 分类:

mybatis(2) 

 最近做一个批量导入的需求,将多条记录批量插入数据库中。解决思路:在程序中封装一个List集合对象,然后把该集合中的实体插入到数据库中,因为项目使用了MyBatis,所以打算使用MyBatis的foreach功能进行批量插入。期间遇到了“SQL 命令未正确结束 ”的错误,最终解决,记录下来供以后查阅和学习。

首先,在网上参考了有关Mybatis的foreach insert的资料具体如下:

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

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

Ø item表示集合中每一个元素进行迭代时的别名,

Ø index指定一个名字,用于表示在迭代过程中,每次迭代到的位置,

Ø open表示该语句以什么开始,

Ø separator表示在每次进行迭代之间以什么符号作为分隔符,

Ø close表示以什么结束,

在使用foreach的时候最关键的也是最容易出错的就是collection属性,该属性是必须指定的,但是在不同情况 下,该属性的值是不一样的,主要有一下3种情况:

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

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

3. 如果传入的参数是多个的时候,我们就需要把它们封装成一个Map了,当然单参数也可以封装成map

 

然后,照葫芦画瓢写了如下的xml文件,

xxxMapper.xml文件:

<insert id="addSupCity" parameterType="java.util.List">

    <selectKey keyProperty="cityId" order="BEFORE" resultType="String">

        <![CDATA[SELECT SEQ_OCL_SUPCITY.NEXTVAL FROM dual]]>

    </selectKey>

    INSERT INTO T_OCL_SUPCITY

    (CITY_ID,CITY_CODE, CITY_NAME, AREA_DESC, SUP_ID, STAT)

    VALUES

    <foreach collection="list" item="item" index="index" separator=",">     

      (

        #{item.cityId,jdbcType=VARCHAR},

        #{item.cityCode,jdbcType=VARCHAR},

        #{item.cityName,jdbcType=VARCHAR},

        #{item.areaDesc,jdbcType=VARCHAR},

        #{item.supId,jdbcType=VARCHAR},

        #{item.stat,jdbcType=VARCHAR}

      )

    </foreach>

</insert>

        但是运行起来后就一直报错,报错信息如下:

### SQL: INSERT INTO T_OCL_SUPCITY

(CITY_ID,CITY_CODE, CITY_NAME, AREA_DESC, SUP_ID, STAT) VALUES (?,?,?,?,?),(?,?,?,?,?)

### Cause: java.sql.SQLSyntaxErrorException: ORA-00933: SQL 命令未正确结束

        把SQL复制出来在PL/SQL中运行也是报同样的错,如上也可以看出,使用批量插入执行的SQL语句等价于: INSERT INTO T_OCL_SUPCITY (CITY_ID,CITY_CODE, CITY_NAME, AREA_DESC, SUP_ID, STAT) VALUES (?,?,?,?,?),(?,?,?,?,?),而在oracle中用insert into xxx values (xxx,xxx),(xxx,xxx) 这种语法是通不过的 。

再回过头去看那篇文章,发现这是适用于MySQL的,不适用于Oracle,因此把xml文件修改一下:

<insert id="addSupCity" parameterType="java.util.List">

      INSERT INTO T_OCL_SUPCITY

  (CITY_ID,CITY_CODE, CITY_NAME, AREA_DESC, SUP_ID, STAT)

SELECT SEQ_OCL_SUPCITY.NEXTVAL CITY_ID, A.*

FROM(

<foreach collection="list" item="item" index="index" separator="UNION ALL">

 SELECT

       #{item.cityCode,jdbcType=VARCHAR} CITY_CODE,

       #{item.cityName,jdbcType=VARCHAR} CITY_NAME,

       #{item.areaDesc,jdbcType=VARCHAR} AREA_DESC,

       #{item.supId,jdbcType=VARCHAR} SUP_ID,

       #{item.stat,jdbcType=VARCHAR} STAT

     FROM dual

   </foreach>

   )A

  </insert>

        运行通过。在Oracle的版本中,有几点需要注意的:

1. SQL中没有VALUES;

2. <foreach>标签中的(selece column from dual);

        3. <foreach>标签中的separator的属性为"UNION ALL",将查询合并结果集。

4. 批量插入的思想宗旨是:插入一张临时表,这张临时表由即将批量插入的数据组成,字段与要最终插入的表一致。

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

SQL UNION 和 UNION ALL 操作符

SQL UNION 操作符

UNION 操作符用于合并两个或多个 SELECT 语句的结果集。

请注意,1. UNION 内部的 SELECT 语句必须拥有相同数量的列。

2. 列也必须拥有相似的数据类型

3. 同时,每条 SELECT 语句中的列的顺序必须相同。

SQL UNION 语法

SELECT column_name(s) FROM table_name1

UNION

SELECT column_name(s) FROM table_name2

注释默认地,UNION 操作符选取不同的值。如果允许重复的值,请使用 UNION ALL。

 

SQL UNION ALL 语法

SELECT column_name(s) FROM table_name1

UNION ALL

SELECT column_name(s) FROM table_name2

另外,UNION 结果集中的列名总是等于 UNION 中第一个 SELECT 语句中的列名。

 

下面的例子中使用的原始表

Employees_China:

E_ID

E_Name

01

Zhang, Hua

02

Wang, Wei

03

Carter, Thomas

04

Yang, Ming

Employees_USA:

E_ID

E_Name

01

Adams, John

02

Bush, George

03

Carter, Thomas

04

Gates, Bill

使用 UNION 命令

实例

列出所有在中国和美国的不同的雇员名

SELECT E_Name FROM Employees_China

UNION

SELECT E_Name FROM Employees_USA

结果

E_Name

Zhang, Hua

Wang, Wei

Carter, Thomas

Yang, Ming

Adams, John

Bush, George

Gates, Bill

注释这个命令无法列出在中国和美国的所有雇员。在上面的例子中,我们有两个名字相同的雇员,他们当中只有一个人被列出来了。UNION 命令只会选取不同的值。

UNION ALL

UNION ALL 命令和 UNION 命令几乎是等效的,不过 UNION ALL 命令会列出所有的值。

SQL Statement 1

UNION ALL

SQL Statement 2

使用 UNION ALL 命令

【注意】

使用UNION ALL连接的两张表一定是现在(当前)查询的,而不能是别名,也就是说连接的前后两张表必须要有SELECT关键字开启

语法SELECT … FROM …

UNION ALL

SELECT … FROM …

实例

列出在中国和美国的所有的雇员

SELECT E_Name FROM Employees_China

UNION ALL

SELECT E_Name FROM Employees_USA

结果

E_Name

Zhang, Hua

Wang, Wei

Carter, Thomas

Yang, Ming

Adams, John

Bush, George

Carter, Thomas

Gates, Bill

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值