MyBatis在SQL語(yǔ)句中如何獲取list的大小
MyBatis在SQL語(yǔ)句中獲取list大小
需求:使用MyBatis進(jìn)行開發(fā)時(shí),在一個(gè)SQL語(yǔ)句中需要拼接list的大小。
大家都知道,當(dāng)我們?cè)贛yBatis中寫SQL時(shí),如果需要遍歷list,先對(duì)list進(jìn)行非空判斷的時(shí)候,可以加下面這行:
<if test="null != list and list.size != 0">SQL</if>
但是如果想在SQL中取到list.size的值,則比較麻煩。
一般會(huì)想到如下:
? ? <select id="getArticleIdByTags" parameterType="java.util.List" resultType="java.lang.Integer">
? ? ? ? SELECT id
? ? ? ? FROM table_name
? ? ? ? where 1=1
? ? ? ? <foreach collection="list" item="item" open="AND" separator="OR" close="GROUP BY id HAVING COUNT(*) >=">
? ? ? ? ? ? field1 = #{item.a} AND field2 = #{item.b}
? ? ? ? </foreach>
? ? ? ? #{list.size}
? ? </select>但是你會(huì)發(fā)現(xiàn)報(bào)錯(cuò)了,PersistenceException、UnsupportOperationException。
因?yàn)閘ist.size用#{}是取不出來(lái)的,需要用${},把#換成$就可以了。
#{}能夠有效防止SQL注入,但是也有它的缺點(diǎn),它會(huì)把傳入的數(shù)據(jù)自動(dòng)加上一個(gè)雙引號(hào),所以如果要的是數(shù)字的話,就會(huì)比較尷尬。
而${}可以直接解析出原本的數(shù)據(jù),所以需要數(shù)值比較的話,還是要加${}。
MyBatis SQL語(yǔ)句解讀
1 CRUD標(biāo)簽
1.1 select
select中屬性說(shuō)明:
id屬性:當(dāng)前名稱空間下的statement的唯一標(biāo)識(shí)。必須。要求id和mapper接口中的方法的名字一致。resultType:將結(jié)果集映射為java的對(duì)象類型。必須(和 resultMap 二選一)parameterType:傳入?yún)?shù)類型??梢允÷?/li>
標(biāo)簽內(nèi)部:具體的sql語(yǔ)句。
?? ??? ?<select id="接口名稱" parameterType="包名.類名" resultType="包名.返回結(jié)果集類型"> ?? ??? ??? ?select 字段名 from 表名 where 條件? ?? ??? ?</select>
注意:resultType中填寫返回結(jié)果集中對(duì)象的類型,而不是結(jié)果集的類型
例如:
返回結(jié)果是一個(gè)集合---> List<User>
resultType里面應(yīng)該填寫User ---> resultType="包名.User"
1.2 insert
insert中屬性說(shuō)明:
id:唯一標(biāo)識(shí),必須,在同一個(gè)命名空間下保持唯一,使用動(dòng)態(tài)代理之后要求和方法名保持一致parameterType:參數(shù)的類型,使用動(dòng)態(tài)代理之后和方法的參數(shù)類型一致useGeneratedKeys:開啟主鍵回寫keyColumn:指定數(shù)據(jù)庫(kù)的主鍵keyProperty:主鍵對(duì)應(yīng)的實(shí)體類屬性名
標(biāo)簽內(nèi)部:具體的sql語(yǔ)句。
<insert id="接口名稱" parameterType="包名.類名" useGeneratedKeys="true" keyProperty="對(duì)象屬性"> ? ? ? ? insert into student(字段名,字段名,字段名,字段名) values(參數(shù),參數(shù),參數(shù),參數(shù)); </insert>
注意:keyProperty 返回的主鍵值會(huì)寫入傳入?yún)?shù)的對(duì)象中,所以其中填寫對(duì)象接收主鍵值的屬性
1.3 update
update中屬性說(shuō)明
id屬性:當(dāng)前名稱空間下的statement的唯一標(biāo)識(shí),必須,使用動(dòng)態(tài)代理之后要求和方法名保持一致;parameterType:傳入的參數(shù)類型,可以省略。
標(biāo)簽內(nèi)部:具體的sql語(yǔ)句。
?? ??? ?<update id="接口名稱" parameterType="包名.類名"> ?? ??? ??? ?update 表名 set 字段名=參數(shù)... ?where 條件 ?? ??? ?</update>
1.4 delete
delete 的幾個(gè)屬性說(shuō)明:
id屬性:當(dāng)前名稱空間下的statement的唯一標(biāo)識(shí),必須,使用動(dòng)態(tài)代理之后要求和方法名保持一致;parameterType:傳入的參數(shù)類型,可以省略。
標(biāo)簽內(nèi)部:具體的sql語(yǔ)句。
?? ??? ??? ??? ?<delete id="delete" > ?? ??? ??? ??? ??? ?delete from 表名 where 條件 ?? ??? ??? ??? ?</delete>
1.5 #{ }和${ }的區(qū)別
#{} 只是替換?,相當(dāng)于PreparedStatement使用占位符去替換參數(shù),可以防止sql注入。
${} 是進(jìn)行字符串拼接,相當(dāng)于sql語(yǔ)句中的Statement,使用字符串去拼接sql;${} 可以是sql中的任一部分傳入到Statement中,不能防止sql注入
簡(jiǎn)單說(shuō)可以理解為:
#{} 相當(dāng)于對(duì)數(shù)據(jù)加上了引號(hào),預(yù)編譯的時(shí)候相當(dāng)于占位符號(hào),能夠防止sql注入。
${} 沒(méi)有給數(shù)據(jù)加上引號(hào),預(yù)編譯是相當(dāng)于字符串拼接,所以不能防止sql注入。
2 動(dòng)態(tài)SQL
2.1 if 標(biāo)簽
?? ??? ??? ??? ??? ?<if test=" id != 0 ">
?? ??? ??? ??? ??? ? ? ? ?and id = #{id}
?? ??? ??? ??? ??? ?</if>
?? ??? ??? ??? ??? ?<if test=" name!=null and name.trim()!='' ">
?? ??? ??? ??? ??? ? ? ? ?and name = #{name}
?? ??? ??? ??? ??? ?</if>test是if標(biāo)簽的唯一屬性,它用來(lái)判斷后面的條件是否成立。如果條件成立,就將if標(biāo)簽內(nèi)的內(nèi)容拼接到sql語(yǔ)句中
多個(gè)if連接,只要自身?xiàng)l件成立就會(huì)實(shí)現(xiàn)sql語(yǔ)句拼接,和其他if標(biāo)簽無(wú)關(guān)
2.2 choose when otherwise
?? ??? ??? ??? ?<choose>
?? ??? ??? ??? ? ? ? ? ?<when test="name!=null and name.trim()!=''">
?? ??? ??? ??? ? ? ? ? ? ? ?and name = #{name}
?? ??? ??? ??? ? ? ? ? ?</when>
?? ??? ??? ??? ? ? ? ? ?<when test="id != 0">
?? ??? ??? ??? ? ? ? ? ? ? ?and age = #{age}
?? ??? ??? ??? ? ? ? ? ?</when>
?? ??? ??? ??? ? ? ? ? ?<otherwise>
?? ??? ??? ??? ? ? ? ? ? ? ?and name='鵬程'
?? ??? ??? ??? ? ? ? ? ?</otherwise>
?? ??? ??? ??? ?</choose>choose when otherwise 三個(gè)標(biāo)簽通常一起使用。
作用:
1.一旦有條件成立的when,后續(xù)的when則不會(huì)執(zhí)行
2.當(dāng)所有的when都不執(zhí)行時(shí),才會(huì)執(zhí)行otherwise
2.3 where set
<where> ?</where> <set> ?</set>
**作用:
where標(biāo)簽相當(dāng)于在sql語(yǔ)句中拼接一個(gè)where單詞,里面可以填寫條件
set標(biāo)簽相當(dāng)于在sql語(yǔ)句中拼接一個(gè)set單詞,當(dāng)對(duì)數(shù)據(jù)修改時(shí)可以修改,里面也可以填寫部分語(yǔ)句
兩個(gè)標(biāo)簽都沒(méi)有屬性,就是直接在語(yǔ)句中拼接上where或set
**
2.4 trim
?? ?update student
? ? <trim prefix="set" suffix="" prefixOverrides="" suffixOverrides=",">
? ? ? ? <if test="name!=null and name.trim()!=''">name = #{name},</if>
? ? ? ? <if test="password!=null and password.trim()!=''">password = #{password},</if>
? ? ? ? <if test="sex!=null and sex.trim()!=''">sex = #{sex},</if>
? ? ? ? <if test="age!=0">age = #{age},</if>
? ? ? ? <if test="tel!=null and tel.trim()!=''">tel = #{tel},</if>
? ? ? ? <if test="address!=null and address.trim()!=''">address = #{address},</if>
? ? ?</trim>
? ? ?<if test="id!=0">where id = #{id}</if>| 屬性 | 作用 |
|---|---|
| prefix | 添加前綴 |
| suffix | 添加后綴 |
| prefixOverrides | 刪除多余前綴 |
| suffixOverrides | 刪除多余后綴 |
2.5 foreach
?? ??? ? select * from tb_user where id in
?? ??? ? <foreach collection="ids" item="id" open="(" close=")" separator=",">
?? ??? ? ? ? ? #{id}
?? ??? ? </foreach>foreach標(biāo)簽可以遍歷集合。所以我們?cè)谑褂胒oreach時(shí),傳入的參數(shù)都是集合。
其中各元素說(shuō)明:
collection:傳入集合參數(shù)名稱id:循環(huán)對(duì)象名稱,自己設(shè)置open:開始符號(hào),與close一起使用,就是循環(huán)輸出的結(jié)果外面加上所需要的符號(hào)close:結(jié)束符號(hào)separator:設(shè)置迭代元素之間的間隔符
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringMVC中Json數(shù)據(jù)交互處理示例詳解
這篇文章主要介紹了SpringMVC中Json數(shù)據(jù)交互處理的相關(guān)資料,分別講解了JSON的基本概念、構(gòu)成要素、數(shù)據(jù)類型、對(duì)象和數(shù)組的表示方法、字符串的轉(zhuǎn)義規(guī)則以及JSON與JavaScript的關(guān)系,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2025-03-03
Java多線程編程基石ThreadPoolExecutor示例詳解
這篇文章主要為大家介紹了Java多線程編程基石ThreadPoolExecutor示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04
關(guān)于SpringSecurity?Context?中獲取和更改當(dāng)前用戶信息的問(wèn)題
SpringSecurityContext在異步線程中無(wú)法獲取用戶信息,因其與請(qǐng)求線程綁定;此外,用戶信息更新后跳轉(zhuǎn)頁(yè)面時(shí),身份會(huì)被降級(jí)為匿名,導(dǎo)致信息無(wú)法及時(shí)同步,本文給大家介紹SpringSecurity?Context?中獲取和更改當(dāng)前用戶信息的問(wèn)題,感興趣的朋友一起看看吧2024-09-09
新版Android?Studio修改jdk版本的簡(jiǎn)單步驟
這篇文章主要介紹了新版Android?Studio修改jdk版本的簡(jiǎn)單步驟,升級(jí)Android?Studio后,JDK版本設(shè)置選項(xiàng)消失,可以通過(guò)Settings面板進(jìn)入Gradle設(shè)置,修改GradleJDK路徑來(lái)解決問(wèn)題,需要的朋友可以參考下2025-03-03
springboot jsp支持以及轉(zhuǎn)發(fā)配置方式
文章介紹了如何在Spring Boot項(xiàng)目中配置和使用JSP,并提供了一步一步的指導(dǎo),包括添加依賴、配置文件設(shè)置、控制器和視圖的使用2024-12-12
使用list stream: 任意對(duì)象List拼接字符串
這篇文章主要介紹了使用list stream:任意對(duì)象List拼接字符串操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
Java?設(shè)計(jì)模式以虹貓藍(lán)兔的故事講解簡(jiǎn)單工廠模式
簡(jiǎn)單工廠模式是屬于創(chuàng)建型模式,又叫做靜態(tài)工廠方法(Static Factory Method)模式,但不屬于23種GOF設(shè)計(jì)模式之一。簡(jiǎn)單工廠模式是由一個(gè)工廠對(duì)象決定創(chuàng)建出哪一種產(chǎn)品類的實(shí)例。簡(jiǎn)單工廠模式是工廠模式家族中最簡(jiǎn)單實(shí)用的模式,可以理解為是不同工廠模式的一個(gè)特殊實(shí)現(xiàn)2022-03-03
SpringBoot使用Redis實(shí)現(xiàn)消息隊(duì)列的方法小結(jié)
在應(yīng)用中把Redis當(dāng)成消息隊(duì)列來(lái)使用已經(jīng)屢見不鮮了,我想主要原因是當(dāng)代應(yīng)用十有八九都會(huì)用到 Redis,因此不用再引入其他消息隊(duì)列系統(tǒng),而且Redis提供了好幾種實(shí)現(xiàn)消息隊(duì)列的方法,用起來(lái)也簡(jiǎn)單,本文給大家介紹了SpringBoot使用Redis實(shí)現(xiàn)消息隊(duì)列的方法小結(jié)2024-04-04

