溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

mybatis中foreach collection的示例分析

發(fā)布時間:2021-08-25 10:01:45 來源:億速云 閱讀:238 作者:小新 欄目:編程語言

這篇文章主要介紹mybatis中foreach collection的示例分析,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!

在SQL開發(fā)過程中,動態(tài)構建In集合條件查詢是比較常見的用法,在Mybatis中提供了foreach功能,該功能比較強大,它允許你指定一個集合,聲明集合項和索引變量,它們可以用在元素體內。它也允許你指定開放和關閉的字符串,在迭代之間放置分隔符。這個元素是很智能的,它不會偶然地附加多余的分隔符。

下面是一個演示示例:

 <select id="findByIdsMap" resultMap="BaseResultMap">  
 Select  
 <include refid="Base_Column_List" />  
 from jria where ID in   <foreach item="item" index="index" collection="list"          
 open="(" separator="," close=")">         
 #{item}      
 </foreach> 
</select>

    但由于官方文檔對這塊的使用,描述的比較簡短,細節(jié)上也被忽略掉了(可能是開源項目文檔一貫的問題吧),也使用不少同學在使用中遇到了問題。特別是foreach這個函數(shù)中,collection屬性做什么用,有什么注意事項。由于文檔不全,這塊只能通過源代碼剖析的方式來分析一下各個屬性的相關要求。

collection屬性的用途是接收輸入的數(shù)組或是List接口實現(xiàn)。但對于其名稱的要求,Mybatis在實現(xiàn)中還是有點不好理解的,所以需要特別注意這一點。

下面開始分析源代碼(筆記使用的是Mybatis 3.0.5版本)

先找到Mybatis執(zhí)行SQL配置解析的入口

MapperMethod.Java類中 public Object execute(Object[] args) 該方法是執(zhí)行的入口.

針對in集合查詢,對應用就是 selectForList或SelctForMap方法。

mybatis中foreach collection的示例分析

但不管調用哪個方法,都會對原來JDK傳入的參數(shù) Object[]類型,通過 getParam方法轉換成一個Object,那這個方法是做什么的呢?分析源碼如下:

mybatis中foreach collection的示例分析

上圖中標紅的兩處,很驚訝的發(fā)現(xiàn),一個參數(shù)與多個參數(shù)的處理方式是不同的(后續(xù)很多同學遇到的問題,就有一大部分出自這個地方)。如果參數(shù)個數(shù)大于一個,則會被封裝成Map, key值如果使用了Mybatis的 Param注解,則會使用該key值,否則默認統(tǒng)一使用數(shù)據(jù)序號,從1開始。這個問題先記下,繼續(xù)分析代碼,接下來如果是selectForList操作(其它操作就對應用相應方法),會調用DefaultSqlSession的public List selectList(String statement, Object parameter, RowBounds rowBounds) 方法

又一個發(fā)現(xiàn),見源代碼如下:

mybatis中foreach collection的示例分析

上圖標紅部分,對參數(shù)又做了一次封裝,我們看一下代碼

mybatis中foreach collection的示例分析

現(xiàn)在有點清楚了,如果參數(shù)類型是List,則必須在collecion中指定為list, 如果是數(shù)據(jù)組,則必須在collection屬性中指定為 array.

現(xiàn)在就問題就比較清楚了,如果是一個參數(shù)的話,collection的值取決于你的參數(shù)類型。

如果是多個值的話,除非使用注解Param指定,否則都是數(shù)字開頭,所以在collection中指定什么值都是無用的。下圖是debug顯示結果。

mybatis中foreach collection的示例分析

針對上面分析的結果,下面給出了一個使用的解決方案,希望對大家對幫助。

在使用這個功能是需要特別注意以下規(guī)則:

1. 當查詢的參數(shù)只有一個時 

findByIds(List<Long> ids)
1.a 如果參數(shù)的類型是List, 則在使用時,collection屬性要必須指定為 
list <select id="findByIdsMap" resultMap="BaseResultMap">     
Select     
<include refid="Base_Column_List" />    
from jria where ID in
         <foreach item="item" index="index" collection="list" 
             open="(" separator="," close=")">
            #{item}
        </foreach>
 </select>  
 findByIds(Long[] ids) 1.b 如果參數(shù)的類型是Array,則在使用時,collection屬性要必須指定為 array <select id="findByIdsMap" resultMap="BaseResultMap">
         select
         <include refid="Base_Column_List" />
     from jria where ID in
         <foreach item="item" index="index" collection="array" 
             open="(" separator="," close=")">
            #{item}
        </foreach>
 </select>

  2. 當查詢的參數(shù)有多個時,例如 findByIds(String name, Long[] ids) 這種情況需要特別注意,在傳參數(shù)時,一定要改用Map方式, 這樣在collection屬性可以指定名稱            

下面是一個示例    

 Map<String, Object> params = new HashMap<String, Object>(2);    params.put("name", name);     params.put("ids", ids);    mapper.findByIdsMap(params); <select id="findByIdsMap" resultMap="BaseResultMap">
         select
         <include refid="Base_Column_List" />
     from jria where ID in
         <foreach item="item" index="index" collection="ids" 
             open="(" separator="," close=")">
            #{item}
        </foreach>
  </select>

   完整的示例如下:例如有一個查詢功能,Mapper接口文件定義如下方法: 

 List<Jria> findByIds(Long... ids);

   使用 in 查詢的sql拼裝方法如下:  

 <select id="findbyIds" resultMap="BaseResultMap">
         select
         <include refid="Base_Column_List" />
     from jria where ID in
         <foreach item="item" index="index" collection="array" 
             open="(" separator="," close=")">
            #{item}
        </foreach>
 </select>

以上是“mybatis中foreach collection的示例分析”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業(yè)資訊頻道!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經查實,將立刻刪除涉嫌侵權內容。

AI