技术前提:我们使用iBatis作为持久层方案

技术场景: 
    假设我们有两张表,一张主表Main,一张子表Sub,并且主表的主键是由数据库维护的自增长的主键,子表中有一个字段引用这个主键,那么当我们插入主表数据后,就需要马上返回这个自增长的主键。

解决方案: 
    可以在insert时通过iBatis的selectKey返回那个主键。 
    selectKey有3个主要的属性: 
    1)resultClass:返回的主键的数据类型,跟sqlMap中的数据类型一致; 
    2)type:表示主键在insert之前或之后生成(取决于数据库的主键生成策略),取值分别为[pre|post],非必须,未填写时如果在insert之前表示pre,否则表示post; 
    2)keyProperty:返回值保存到的属性,非必须(作用参见Oracle配置);

实际配置: 
<!-- Oracle SEQUENCE -->    
<insert id="insertProduct-ORACLE" parameterClass="com.domain.Product">    
    <selectKey resultClass="int" keyProperty="id" type="pre">    
        <![CDATA[ 
          SELECT STOCKIDSEQUENCE.NEXTVAL AS ID FROM DUAL 
        ]]>    
    </selectKey>    
    <![CDATA[ 
    insert into PRODUCT(PRD_ID,PRD_DESCRIPTION) values(#id#,#description#) 
<!-- 此处体现了keyProperty的作用 -->  
    ]]>    
</insert>

<!-- Microsoft SQL Server IDENTITY Column -->    
<insert id="insertProduct-MS-SQL" parameterClass="com.domain.Product">    
    <![CDATA[ 
      insert into PRODUCT (PRD_DESCRIPTION) values(#description#) 
    ]]>    
    <selectKey resultClass="int" keyProperty="id" type="post">    
        <![CDATA[ 
          SELECT @@IDENTITY AS ID 
        ]]>  
        <!-- 该方法不安全 应当用SCOPE_IDENTITY() 但这个函数属于域函数,需要在一个语句块中执行。 -->  
    </selectKey>    
</insert>  
上述MS SQL Server配置随是官网提供的配置,但实际上却恰恰隐患重重!按下述配置,确保获得有效主键。

<!-- Microsoft SQL Server IDENTITY Column improvement-->    
<insert id="insertProduct-MS-SQL" parameterClass="com.domain.Product">    
    <selectKey resultClass="int" keyProperty="id">    
      <![CDATA[ 
        insert into PRODUCT (PRD_DESCRIPTION) values(#description#)  
        SELECT SCOPE_IDENTITY() AS ID 
      ]]>  
    </selectKey>    
</insert>

<!-- MySQL Last Insert Id -->  
<insert id="insertProduct-Mysql" parameterClass="com.domain.Product">  
    insert into PRODUCT(PRD_DESCRIPTION) values(#description#)    
    <selectKey resultClass="int" keyProperty="id">  
        SELECT LAST_INSERT_ID() AS ID 
    </selectKey>  
</insert> 
<!-- 该方法LAST_INSERT_ID()与数据库连接绑定,同属统一会话级别,不会发生上述MS SQL Server的函数问题。 -->

<!-- DB2 Identity Val Local -->  
<insert id="insertProduct-DB2" parameterClass="com.domain.Product"> 
    insert into PRODUCT(PRD_DESCRIPTION) values(#description#) 
    <selectKey resultClass="int" type="post"> 
        VALUES IDENTITY_VAL_LOCAL() 
    </selectKey> 
</insert>

<!-- Sqlite sqlite_sequence -->  
<insert id="insertProduct-Sqlite" parameterClass="com.domain.Product"> 
    insert into PRODUCT(PRD_DESCRIPTION) values(#description#) 
    <selectKey resultClass="string" type="post"> 
        SELECT seq FROM sqlite_sequence WHERE name = 'PRD_DESCRIPTION' 
    </selectKey> 
</insert> 
<!-- 主键字段一定要加下INTEGER PRIMARY KEY AUTOINCREMENT,否则会报找不到 sqlite_sequence 表。 -->

通过以上方式,可以最大程度上确保插入数据的时候获得当前自增主键。

关于iBatis-selectKey的一点笔记的更多相关文章

  1. 关于最小生成树,拓扑排序、强连通分量、割点、2-SAT的一点笔记

    关于最小生成树,拓扑排序.强连通分量.割点.2-SAT的一点笔记 前言:近期在复习这些东西,就xjb写一点吧.当然以前也写过,但这次偏重不太一样 MST 最小瓶颈路:u到v最大权值最小的路径.在最小生 ...

  2. [转&精]IO_STACK_LOCATION与IRP的一点笔记

    IO_STACK_LOCATION和IRP算是驱动中两个很基础的东西,为了理解这两个东西,找了一点资料. 1. IRP可以看成是Win32窗口程序中的消息(Message),DEVICE_OBJECT ...

  3. 关于SS的一点笔记

    过年的时候抽了点时间了解了下ss的协议.整理了一点笔记,一直没有时间发.今天发一下,免得忘了. SS的结构本身比较简单,他的基本结构如下: ss通常分为client和server两部分 client是 ...

  4. ibatis selectKey

    <insert id="insert" parameterClass="A"> <selectKey keyProperty="uu ...

  5. ibatis selectKey用法问题

    其实就是相为SHIPMENT_HISTORY表加入一个主键sequence id shipmentHistoryId,加入一条记录,然后返回这个sequence id xml 代码 <inser ...

  6. pyhon的yileld的一点笔记

    yield感觉很神秘,感觉也不好理解,学习pyhon最后终归是要学习这个东西,研究了一段时间,把自己的笔记写下来 说简单点就是遇到yield就停止往下执行代码,也包括不执行yield这条语句,然后返回 ...

  7. 关于在VB.NET中调用使用VC++编写的类库dll的一点笔记

    前言 结对作业要求一出来,我就立刻想到了把“计算核心”封装成dll,然后使用vb.net编写UI调用dll的思路.然而在实现过程中却遇到了很多的问题. 我在这个过程中是负责使用vb.net编写UI并调 ...

  8. 阅读xtrabackup代码的一点笔记

    xtrabackup binary最重要的两个过程是backup和prepare,对应的函数分别是xtrabackup_backup_func()和xtrabackup_prepare_func(), ...

  9. c#委托----我的一点笔记

    public partial class ucBloodLabs { public delegate void ShowBloodEvent(); public ShowBloodEvent Show ...

随机推荐

  1. lintcode-86-二叉查找树迭代器

    86-二叉查找树迭代器 设计实现一个带有下列属性的二叉查找树的迭代器: 元素按照递增的顺序被访问(比如中序遍历) next()和hasNext()的询问操作要求均摊时间复杂度是O(1) 样例 对于下列 ...

  2. C++中范围for语句

    如果想对string对象中的每个字符做点什么操作,目前最好的办法是使用C++11新标准提供的一种语句:范围for(range for)语句. 示例代码: #include<iostream> ...

  3. 手动修改PHP页面返回状态码

    <?php //比如当前页面要返回404状态码 header("HTTP/1.1 404 Not Found"); header("Status: 404 Not ...

  4. CFS/FQ/PQ调度与WRR负载均衡

    动机 五一临近,四月也接近尾声,五一节乃小长假的最后一天.今天是最后一天工作日,竟然感冒了,半夜里翻来覆去无法安睡,加上窗外大飞机屋里小飞机(也就是蚊子)的骚扰,实在是必须起来做点有意义的事了!    ...

  5. 雅礼集训 Day3 T3 w 解题报告

    w 题目背景 \(\frac 14\)遇到了一道水题,双完全不会做,于是去请教小\(\text{D}\).小\(\text{D}\)看了\(0.607^2\)眼就切掉了这题,嘲讽了\(\frac 14 ...

  6. C&C++——C函数与C++函数相互调用问题

    C C++相互调用 在项目中融合C和C++有时是不可避免的,在调用对方的功能函数的时候,或许会出现这样那样的问题,但只要我的C代码和我的C++代码分别都能成功编译,那其他就不是问题.近来在主程序是C语 ...

  7. BZOJ 3319 黑白树 并查集+线段树

    这这这这这这什么毒瘤题!!!!!!!!!!!!!!!!!!!!!!!!!!!! 卡LCT(优秀的LCT由于是均摊本身就带着2,3的常数在,而且这道题对于LCT标记十分难维护,又得乘上4,5然后就炸了) ...

  8. NEYC 2017 游记

    day 1:  result:    sum_rank: 11   school_rank:1   水题在你高估的时候就已经不水了   sum:有个快速乘类似快速幂:       int ans=0; ...

  9. JS Cookie相关操作

    function setCookie(cookieName, cookieValue, expires) { // 设置Cookie function getCookieName(cookieName ...

  10. yaf学习网站

    http://www.01happy.com/php-yaf-ext-business/