ibatis 動態mapped statement

 動態 Mapped Statement
直接使用 JDBC 一個非常普遍的問題是動態 SQL。使用參數值、參數本身和數據列都
是動態的 SQL,通常非常困難。典型的解決方法是,使用一系列 if-else 條件語句和一連串
討厭的字符串連接。對於這個問題,SQL Map API使用和 mapped statement非常相似的結構,
這裏是一個簡單的例子:
<select id="dynamicGetAccountList"
cacheModel="account-cache"
resultMap="account-result" >
select * from ACCOUNT
<isGreaterThan prepend="and" property="id" compareValue="0">
where ACC_ID = #id#
</isGreaterThan>
order by ACC_LAST_NAME
</select>
 
上面的例子中,根據參數 bean“id”屬性的不同情況,可創建兩個可能的語句。如果參
數“id”大於 0,將創建下面的語句:
select * from ACCOUNT where ACC_ID = ?
或者,如果“id”參數小於等於 0,將創建下面的語句:
select * from ACCOUNT
在更復雜的例子中,動態 Mapped Statement 的用處更明顯。如下面比較複雜的例子:
<statement id="dynamicGetAccountList" resultMap="account-result" >

select * from ACCOUNT
<dynamic prepend="WHERE">
<isNotNull prepend="AND" property="firstName">
(ACC_FIRST_NAME = #firstName#
<isNotNull prepend="OR" property="lastName">
ACC_LAST_NAME = #lastName#
</isNotNull>
)
</isNotNull>
<isNotNull prepend="AND" property="emailAddress">
ACC_EMAIL like #emailAddress#
</isNotNull>
<isGreaterThan prepend="AND" property="id" compareValue="0">
ACC_ID = #id#
</isGreaterThan>
</dynamic>
order by ACC_LAST_NAME

</statement>
根據不同的條件,上面動態的語句可以產生 16 條不同的查詢語句。使用 if-else 結構和
字符串,會產生上百行很亂的代碼。
而使用動態 Statement,和在 SQL 的動態部位周圍插入條件標籤一樣容易。例如:
<statement id="someName" resultMap="account-result" >
select * from ACCOUNT
<dynamic prepend="where">
<isGreaterThan prepend="and" property="id" compareValue="0">
ACC_ID = #id#
</isGreaterThan>
<isNotNull prepend=”and" property="lastName">
ACC_LAST_NAME = #lastName#
</isNotNull>
</dynamic>
order by ACC_LAST_NAME
</statement>
 
上面的例子中,<dynamic>元素劃分出SQL 語句的動態部分。動態部分可以包含任意多
的條件標籤元素,條件標籤決定是否在語句中包含其中的 SQL 代碼。所有的條件標籤元素
將根據傳給動態查詢 Statement 的參數對象的情況來工作。<dynamic>元素和條件元素都有
“prepend”屬性,它是動態 SQL 代碼的一部分,在必要情況下,可以被父元素的“prepend”
屬性覆蓋。上面的例子中,prepend屬性“where”將覆蓋第一個爲“真”的條件元素。這對

於確保生成正確的 SQL 語句是有必要的。例如,在第一個爲“真”的條件元素中,“AND”
是不需要的,事實上,加上它肯定會出錯。以下小節討論不同的條件元素,包括二元條件元
素,一元條件元素和其他動態元素。
二元條件元素
二元條件元素將一個屬性值和一個靜態值或另一個屬性值比較,如果條件爲“真”,元
素體的內容將被包括在查詢 SQL 語句中。  二元條件元素的屬性:
prepend  - 可被覆蓋的 SQL 語句組成部分,添加在語句的前面(可選)
property  - 被比較的屬性(必選)
compareProperty  -  另一個用於和前者比較的屬性(必選或選擇 compareValue)
compareValue  -  用於比較的值(必選或選擇 compareProperty)
比較屬性值和靜態值或另一個屬性值是否相等。
<isEqual>
比較屬性值和靜態值或另一個屬性值是否不相等。
<isNotEqual>

比較屬性值是否大於靜態值或另一個屬性值。
<isGreaterThan>
比較屬性值是否大於等於靜態值或另一個屬性值。
<isGreaterEqual>
比較屬性值是否小於靜態值或另一個屬性值。
<isLessThan>
比較屬性值是否小於等於靜態值或另一個屬性值。
<isLessEqual>
例子:
<isLessEqual prepend=”AND” property=”age” compareValue=”18”>
ADOLESCENT = ‘TRUE’
</isLessEqual>
 
一元條件元素
一元條件元素檢查屬性的狀態是否符合特定的條件。  一元條件元素的屬性:
prepend  - 可被覆蓋的 SQL 語句組成部分,添加在語句的前面(可選)
property  - 被比較的屬性(必選)

檢查是否存在該屬性(存在 parameter bean 的屬性)。
<isPropertyAvailable>
檢查是否不存在該屬性(不存在 parameter bean 的屬性)。
<isNotPropertyAvailable>
檢查屬性是否爲 null。
<isNull>
檢查屬性是否不爲 null。
<isNotNull>
檢查 Collection.size()的值,屬性的 String 或 String.valueOf()值,
<isEmpty>
是否爲 null或空(“”或size() < 1)。
檢查 Collection.size()的值,屬性的 String 或 String.valueOf()值,
<isNotEmpty>
是否不爲 null 或不爲空(“”或 size() > 0)。
例子:
<isNotEmpty prepend=”AND” property=”firstName” >
FIRST_NAME=#firstName#
</isNotEmpty>
 
 
其他元素  Parameter Present:這些元素檢查參數對象是否存在。
Parameter Present的屬性:
prepend  - 可被覆蓋的 SQL 語句組成部分,添加在語句的前面(可選)
檢查是否存在參數對象(不爲 null)。
<isParameterPresent>
檢查是否不存在參數對象(參數對象爲 null)。
<isNotParameterPresent>
例子:
<isNotParameterPresent prepend=”AND”>
EMPLOYEE_TYPE = ‘DEFAULT’
</isNotParameterPresent>
  Iterate:這屬性遍歷整個集合,併爲 List 集合中的元素重複元素體的內容。
Iterate 的屬性:
prepend  - 可被覆蓋的 SQL 語句組成部分,添加在語句的前面(可選)
property  - 類型爲 java.util.List 的用於遍歷的元素(必選)
open  -  整個遍歷內容體開始的字符串,用於定義括號(可選)
close  -整個遍歷內容體結束的字符串,用於定義括號(可選)
conjunction -  每次遍歷內容之間的字符串,用於定義 AND 或 OR(可選)
遍歷類型爲 java.util.List的元素。
<iterate>
例子:
<iterate prepend=”AND” property=”userNameList”
open=”(” close=”)” conjunction=”OR”>
username=#userNameList[]#
</iterate>
 
注意:使用<iterate>時,在List元素名後面包括方括號[]非常重要,方括號[]將
對象標記爲List,以防解析器簡單地將List輸出成String。
發佈了31 篇原創文章 · 獲贊 1 · 訪問量 3萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章