zoukankan      html  css  js  c++  java
  • MyBatis魔法堂:Insert操作详解(返回主键、批量插入)

     

    一、前言                                   

      数据库操作怎能少了INSERT操作呢?下面记录MyBatis关于INSERT操作的笔记,以便日后查阅。

    二、 insert元素 属性详解                            

      其属性如下:

       parameterType ,入参的全限定类名或类型别名

       keyColumn ,设置数据表自动生成的主键名。对特定数据库(如PostgreSQL),若自动生成的主键不是第一个字段则必须设置

       keyProperty ,默认值unset,用于设置getGeneratedKeys方法或selectKey子元素返回值将赋值到领域模型的哪个属性中

       useGeneratedKeys ,取值范围 true|false(默认值),设置是否使用JDBC的getGenereatedKeys方法获取主键并赋值到keyProperty设置的领域模型 属性中。MySQL和SQLServer执行auto-generated key field,因此当数据库设置好自增长主键后,可通过JDBC的getGeneratedKeys方法获取。但像Oralce等不支持auto- generated key field的数据库就不能用这种方法获取主键了

       statementType ,取值范围STATEMENT,PREPARED(默认值),CALLABLE

       flushCache ,取值范围true(默认值)|false,设置执行该操作后是否会清空二级缓存和本地缓存

       timeout ,默认为unset(依赖jdbc驱动器的设置),设置执行该操作的最大时限,超时将抛异常

       databaseId ,取值范围oracle|mysql等,表示数据库厂家,元素内部可通过`<if test="_databaseId = 'oracle'">`来为特定数据库指定不同的sql语句

    三、一般的INSERT操作——返回值为插入的记录数目                

    mapper接口代码:

    /**
     * 添加学生信息
     * @param student 学生实例
     * @return 成功操作的记录数目
     */
    int add(EStudent student);

    mapper.xml:

    <insert id="add" parameterType="EStudent">
      insert into TStudent(name, age) values(#{name}, #{age})
    </insert

    四、执行INSERT操作后获取记录主键                        

    mapper接口代码:

    /**
     * 添加学生信息
     * @param student 学生实例
     * @return 成功操作的记录数目
     */
    int add(EStudent student);

    至于mapper.xml则分为两种情况了,一种是数据库(如MySQL,SQLServer)支持auto-generated key field,另一种是数据库(如Oracle)不支持auto-generated key field的。

     1. 数据库(如MySQL,SQLServer)支持auto-generated key field的情况

        手段①(推荐做法):

    <insert id="add" parameterType="EStudent" useGeneratedKeys="true" keyProperty="id">
      insert into TStudent(name, age) values(#{name}, #{age})
    </insert>

        手段②:

    复制代码
    <insert id="add" parameterType="EStudent">
      // 下面是SQLServer获取最近一次插入记录的主键值的方式
      <selectKey resultType="_long" keyProperty="id" order="AFTER">
        select @@IDENTITY as id
      </selectKey>
      insert into TStudent(name, age) values(#{name}, #{age})
    </insert>
    复制代码

      由于手段②获取主键的方式依赖数据库本身,因此推荐使用手段①。

     2. 数据库(如Oracle)不支持auto-generated key field的情况

    复制代码
    <insert id="add" parameterType="EStudent">
      <selectKey keyProperty="id" resultType="_long" order="BEFORE">
        select CAST(RANDOM * 100000 as INTEGER) a FROM SYSTEM.SYSDUMMY1
      </selectKey>
      insert into TStudent(id, name, age) values(#{id}, #{name}, #{age})
    </insert
    复制代码

      注意:mapper接口返回值依然是成功插入的记录数,但不同的是主键值已经赋值到领域模型实体的id中了。

    五、 selectKey子元素 详解                            

      作用:在insert元素和update元素中插入查询语句。

      其属性如下:

         keyProperty ,默认值unset,用于设置getGeneratedKeys方法或selectKey子元素返回值将赋值到领域模型的哪个属性中

         resultType ,keyPropety所指向的属性类全限定类名或类型别名

         order属性 ,取值范围BEFORE|AFTER,指定是在insert语句前还是后执行selectKey操作

         statementType ,取值范围STATEMENT,PREPARED(默认值),CALLABLE

    注意:selectKey操作会将操作查询结果赋值到insert元素的parameterType的入参实例下对应的属性中。并提供给insert语句使用

    六、批量插入                                 

      方式1:

    <insert id="add" parameterType="EStudent">
      <foreach collection="list" item="item" index="index" separator=";">
        INSERT INTO TStudent(name,age) VALUES(#{item.name}, #{item.age})
      </foreach>
    </insert>

    上述方式相当语句逐条INSERT语句执行,将出现如下问题:
    1. mapper接口的add方法返回值将是最一条INSERT语句的操作成功的记录数目(就是0或1),而不是所有INSERT语句的操作成功的总记录数目
    2. 当其中一条不成功时,不会进行整体回滚。

      方式2(仅限于MSSQL):

    复制代码
    <insert id="add" parameterType="EStudent">
      WITH R AS
      <foreach collection="list" item="item" index="index" open="(" close=")" separator="union all">
        SELECT #{item.name} as a, #{item.age} as b
      </foreach>
      INSERT INTO TStudent(name,age) SELECT a, b FROM R
    </insert>
    复制代码

    上述方式解决了方式1中的问题。但该方式仅限于MSSQL

      方式3(通用解决方法):

     INSERT INTO TStudent(name,age) 
      <foreach collection="list" item="item" index="index" open="(" close=")" separator="union all">
        SELECT #{item.name} as a, #{item.age} as b
      </foreach>

    该方式与方式2效果一样,而且不仅限于MSSQL

  • 相关阅读:
    【转】VS2010中 C++创建DLL图解
    [转]error: 'retainCount' is unavailable: not available in automatic reference counting mode
    [转]关于NSAutoreleasePool' is unavailable: not available in automatic reference counting mode的解决方法
    【转】 Tomcat v7.0 Server at localhost was unable to start within 45
    【转】Server Tomcat v7.0 Server at localhost was unable to start within 45 seconds. If
    【转】SVN管理多个项目版本库
    【转】eclipse安装SVN插件的两种方法
    【转】MYSQL启用日志,和查看日志
    【转】Repository has not been enabled to accept revision propchanges
    【转】SVN库的迁移
  • 原文地址:https://www.cnblogs.com/deepbreath/p/4447819.html
Copyright © 2011-2022 走看看