ibatis經常使用16條SQL語句


 

(1) 輸入參數爲單個值sql

  1. <delete id="com.fashionfree.stat.accesslog.deleteMemberAccessLogsBefore"   
    parameterClass="long">   
    delete from   
    MemberAccessLog   
    where   
    accessTimestamp = #value#   
    </delete>

 
(2) 輸入參數爲一個對象
數據庫

 

  1. <insert id="com.fashionfree.stat.accesslog.MemberAccessLog.insert"   
    parameterClass="com.fashionfree.stat.accesslog.model.MemberAccessLog>   
    insert into MemberAccessLog   
    (   
    accessLogId, memberId, clientIP,   
    httpMethod, actionId, requestURL,   
    accessTimestamp, extend1, extend2,   
    extend3   
    )   
    values   
    (   
    #accessLogId#, #memberId#,   
    #clientIP#, #httpMethod#,   
    #actionId#, #requestURL#,   
    #accessTimestamp#, #extend1#,   
    #extend2#, #extend3#   
    )   
    </insert>

 

(3) 輸入參數爲一個java.util.HashMap 數組


  1. <select id="com.fashionfree.stat.accesslog.selectActionIdAndActionNumber"   
    parameterClass="hashMap"   
    resultMap="getActionIdAndActionNumber">   
    select   
    actionId, count(*) as count   
    from   
    MemberAccessLog   
    where   
    memberId = #memberId#   
    and accessTimestamp &gt; #start#   
    and accessTimestamp &lt;= #end#   
    group by actionId   
    </select>

 

  (4) 輸入參數中含有數組緩存

   

  1. <insert id="updateStatusBatch" parameterClass="hashMap">   
    update   
    Question   
    set   
    status = #status#   
    <dynamic prepend="where questionId in">   
    <isNotNull property="actionIds">   
    <iterate property="actionIds" open="(" close=")" conjunction=",">   
    #actionIds[]#   
    </iterate>   
    </isNotNull>   
    </dynamic>   
    </insert>

   說明:actionIds爲傳入的數組的名字; 
   使用dynamic標籤避免數組爲空時致使sql語句語法出錯; 
   使用isNotNull標籤避免數組爲null時ibatis解析出錯app

 

   (5)傳遞參數只含有一個數組 
  框架

  1. <select id="com.fashionfree.stat.accesslog.model.StatMemberAction.selectActionIdsOfModule"   
    resultClass="hashMap">   
    select   
    moduleId, actionId   
    from   
    StatMemberAction   
    <dynamic prepend="where moduleId in">   
    <iterate open="(" close=")" conjunction=",">   
    #[]#   
    </iterate>   
    </dynamic>   
    order by   
    moduleId   
    </select>

    說明:注意select的標籤中沒有parameterClass一項 
       另:這裏也能夠把數組放進一個hashMap中,但增長額外開銷,不建議使用dom

 

   (6)讓ibatis把參數直接解析成字符串 
  函數

  1. <select id="com.fashionfree.stat.accesslog.selectSumDistinctCountOfAccessMemberNum"   
    parameterClass="hashMap" resultClass="int">   
    select   
    count(distinct memberId)   
    from   
    MemberAccessLog   
    where   
    accessTimestamp &gt;= #start#   
    and accessTimestamp &lt; #end#   
    and actionId in $actionIdString$   
    </select>

    說明:使用這種方法存在sql注入的風險,不推薦使用spa

 

    (7)分頁查詢 (pagedQuery)

   

  1. <select id="com.fashionfree.stat.accesslog.selectMemberAccessLogBy"   
    parameterClass="hashMap" resultMap="MemberAccessLogMap">   
    <include refid="selectAllSql"/>   
    <include refid="whereSql"/>   
    <include refid="pageSql"/>   
    </select>   
    <select id="com.fashionfree.stat.accesslog.selectMemberAccessLogBy.Count"   
    parameterClass="hashMap" resultClass="int">   
    <include refid="countSql"/>   
    <include refid="whereSql"/>   
    </select>   
    <sql id="selectAllSql">   
    select   
    accessLogId, memberId, clientIP,   
    httpMethod, actionId, requestURL,   
    accessTimestamp, extend1, extend2,   
    extend3   
    from   
    MemberAccessLog   
    </sql>   
    <sql id="whereSql">   
    accessTimestamp &lt;= #accessTimestamp#   
    </sql>   
    <sql id="countSql">   
    select   
    count(*)   
    from   
    MemberAccessLog   
    </sql>   
    <sql id="pageSql">   
    <dynamic>   
    <isNotNull property="startIndex">   
    <isNotNull property="pageSize">   
    limit #startIndex# , #pageSize#   
    </isNotNull>   
    </isNotNull>   
    </dynamic>   
    </sql>

   說明:本例中,代碼應爲: 
   HashMap hashMap = new HashMap(); 
   hashMap.put(「accessTimestamp」, someValue); 
   pagedQuery(「com.fashionfree.stat.accesslog.selectMemberAccessLogBy」, hashMap); 
   pagedQuery方法首先去查找名爲com.fashionfree.stat.accesslog.selectMemberAccessLogBy.Count 的mapped statement來進行sql查詢,從而獲得com.fashionfree.stat.accesslog.selectMemberAccessLogBy查詢的記錄個數, 
再進行所需的paged sql查詢(com.fashionfree.stat.accesslog.selectMemberAccessLogBy),具體過程參見utils類中的相關代碼


(8)sql語句中含有大於號>、小於號<

    1. 將大於號、小於號寫爲: &gt; &lt; 如: 

<delete id="com.fashionfree.stat.accesslog.deleteMemberAccessLogsBefore" parameterClass="long">   
delete from   
MemberAccessLog   
where   
accessTimestamp &lt;= #value#   
</delete>   
 
    2. 將特殊字符放在xml的CDATA區內: 
Xml代碼  
<delete id="com.fashionfree.stat.accesslog.deleteMemberAccessLogsBefore" parameterClass="long">   
<![CDATA[  
delete from  
MemberAccessLog  
where  
accessTimestamp <= #value#  
]]>   
</delete>

   推薦使用第一種方式,寫爲&lt; 和 &gt; (XML不對CDATA裏的內容進行解析,所以若是CDATA中含有dynamic標籤,將不起做用) 

(9)include和sql標籤 
   將經常使用的sql語句整理在一塊兒,便於共用: 

  1. <sql id="selectBasicSql">   
    select   
    samplingTimestamp,onlineNum,year,   
    month,week,day,hour   
    from   
    OnlineMemberNum   
    </sql>   
    <sql id="whereSqlBefore">   
    where samplingTimestamp &lt;= #samplingTimestamp#   
    </sql>   
    <select id="com.fashionfree.accesslog.selectOnlineMemberNumsBeforeSamplingTimestamp" parameterClass="hashmap" resultClass="OnlineMemberNum">   
    <include refid="selectBasicSql" />   
    <include refid="whereSqlBefore" />   
    </select>

    注意:sql標籤只能用於被引用,不能看成mapped statement。如上例中有名爲selectBasicSql的sql元素,試圖使用其做爲sql語句執行是錯誤的: 
    sqlMapClient.queryForList(「selectBasicSql」); ×

(10)隨機選取記錄


  1. <sql id=」randomSql」>   
    ORDER BY rand() LIMIT #number#   
    </sql>

    從數據庫中隨機選取number條記錄(只適用於MySQL)

 

(11)將SQL GROUP BY分組中的字段拼接


  1. <sql id=」selectGroupBy>   
    SELECT   
    a.answererCategoryId, a.answererId, a.answererName,   
    a.questionCategoryId, a.score, a.answeredNum,   
    a.correctNum, a.answerSeconds, a.createdTimestamp,   
    a.lastQuestionApprovedTimestamp, a.lastModified, GROUP_CONCAT(q.categoryName) as categoryName   
    FROM   
    AnswererCategory a, QuestionCategory q   
    WHERE a.questionCategoryId = q.questionCategoryId   
    GROUP BY a.answererId   
    ORDER BY a.answererCategoryId   
    </sql>

    注:SQL中使用了MySQL的GROUP_CONCAT函數

(12) 按照IN裏面的順序進行排序

    ①MySQL: 

  1. <sql id=」groupByInArea」>   
    select   
    moduleId, moduleName,   
    status, lastModifierId, lastModifiedName,   
    lastModified   
    from   
    StatModule   
    where   
    moduleId in (3, 5, 1)   
    order by   
    instr(',3,5,1,' , ','+ltrim(moduleId)+',')   
    </sql>

    

②SQLSERVER:

  1. <sql id=」groupByInArea」>   
    select   
    moduleId, moduleName,   
    status, lastModifierId, lastModifiedName,   
    lastModified   
    from   
    StatModule   
    where   
    moduleId in (3, 5, 1)   
    order by   
    charindex(','+ltrim(moduleId)+',' , ',3,5,1,')   
    </sql>

    說明:查詢結果將按照moduleId在in列表中的順序(3, 5, 1)來返回 
    MySQL : instr(str, substr) 
    SQLSERVER: charindex(substr, str) 
    返回字符串str 中子字符串的第一個出現位置 
    ltrim(str) 
    返回字符串str, 其引導(左面的)空格字符被刪除

(13) resultMap 
    
resultMap負責將SQL查詢結果集的列值映射成Java Bean的屬性值。

  1.  

  2. <resultMap class="java.util.HashMap" id="getActionIdAndActionNumber">   
    <result column="actionId" property="actionId" jdbcType="BIGINT" javaType="long"/>   
    <result column="count" property="count" jdbcType="INT" javaType="int"/>   
    </resultMap>

   使用resultMap稱爲顯式結果映射,與之對應的是resultClass(內聯結果映射),使用resultClass的最大好處即是簡單、方便,不需顯示指定結果,由iBATIS根據反射來肯定自行決定。而resultMap則能夠經過指定jdbcType和javaType,提供更嚴格的配置認證。


(14) typeAlias

  1. <typeAlias alias="MemberOnlineDuration" type="com.fashionfree.stat.accesslog.model.MemberOnlineDuration" />   
    <typeAlias>容許你定義別名,避免重複輸入過長的名字。

 

(15) remap

  1. <select id="testForRemap" parameterClass="hashMap" resultClass="hashMap" remapResults="true">   
    select   
    userId   
    <isEqual property="tag" compareValue="1">   
    , userName   
    </isEqual>   
    <isEqual property="tag" compareValue="2">   
    , userPassword   
    </isEqual>   
    from   
    UserInfo   
    </select>

  此例中,根據參數tag值的不一樣,會得到不一樣的結果集,若是沒有remapResults="true"屬性,iBatis會將第一次查詢時的結果集緩存,下次再執行時(必須仍是該進程中)不會再執行結果集映射,而是會使用緩存的結果集。 
所以,若是上面的例子中remapResult爲默認的false屬性,而有一段程序這樣書寫: 

Java代碼  收藏代碼

  1. HashMap<String, Integer> hashMap = new HashMap<String, Integer>();   
    hashMap.put("tag", 1);   
    sqlClient.queryForList("testForRemap", hashMap);   
    hashMap.put("tag", 2);   
    sqlClient.queryForList("testForRemap", hashMap);

 則程序會在執行最後一句的query查詢時報錯,緣由就是iBATIS使用了第一次查詢時的結果集,而先後兩次的結果集是不一樣的:(userId, userName)和(userId, userPassword),因此致使出錯。若是使用了remapResults="true"這一屬性,iBATIS會在每次執行查詢時都執行結果集映射,從而避免錯誤的發生(此時會有較大的開銷)。 

(16) dynamic標籤的prepend

  dynamic標籤的prepend屬性做爲前綴添加到結果內容前面,當標籤的結果內容爲空時,prepend屬性將不起做用。 
當dynamic標籤中存在prepend屬性時,將會把其嵌套子標籤的第一個prepend屬性忽略。例如:

  1. <sql id="whereSql">   
    <dynamic prepend="where ">   
    <isNotNull property="userId" prepend="BOGUS">   
    userId = #userId#   
    </isNotNull>   
    <isNotEmpty property="userName" prepend="and ">   
    userName = #userName#   
    </isNotEmpty>   
    </dynamic>   
    </sql>

  

此例中,dynamic標籤中含有兩個子標籤<isNotNull>和<isNotEmpty>。根據前面敘述的原則,若是<isNotNull>標籤中沒有prepend="BOGUS" 這一假的屬性來讓dynamic去掉的話,<isNotEmpty>標籤中的and就會被忽略,會形成sql語法錯誤。    注意:當dynamic標籤沒有prepend屬性時,不會自動忽略其子標籤的第一個prepend屬性。 

相關文章
相關標籤/搜索