一、MyBatis動(dòng)態(tài) sql 是什么
動(dòng)態(tài) SQL 是 MyBatis 的強(qiáng)大特性之一。在 JDBC 或其它類似的框架中,開發(fā)人員通常需要手動(dòng)拼接 SQL 語(yǔ)句。根據(jù)不同的條件拼接 SQL 語(yǔ)句是一件極其痛苦的工作。例如,拼接時(shí)要確保添加了必要的空格,還要注意去掉列表最后一個(gè)列名的逗號(hào)。而動(dòng)態(tài) SQL 恰好解決了這一問題,可以根據(jù)場(chǎng)景動(dòng)態(tài)的構(gòu)建查詢。
動(dòng)態(tài)SQL(code that is executed dynamically),它一般是根據(jù)用戶輸入或外部條件動(dòng)態(tài)組合的SQL語(yǔ)句塊。 動(dòng)態(tài)SQL能靈活的發(fā)揮SQL強(qiáng)大的功能、方便的解決一些其它方法難以解決的問題。 相信使用過動(dòng)態(tài)SQL的人都能體會(huì)到它帶來的便利,然而動(dòng)態(tài)SQL有時(shí)候在執(zhí)行性能 (效率)上面不如靜態(tài)SQL,而且使用不恰當(dāng),往往會(huì)在安全方面存在隱患 (SQL 注入式攻擊)。
1.Mybatis 動(dòng)態(tài) sql 是做什么的?
Mybatis 動(dòng)態(tài) sql 可以讓我們?cè)?Xml 映射文件內(nèi),以標(biāo)簽的形式編寫動(dòng)態(tài) sql,完成邏輯判斷和動(dòng)態(tài)拼接 sql 的功能。
2.Mybatis 的 9 種 動(dòng) 態(tài) sql 標(biāo) 簽有哪些?
元素 | 作用 | 備注 |
---|---|---|
if | 判斷語(yǔ)句 | 單條件分支判斷 |
choose(when、otherwise) | 相當(dāng)于 Java 中的 switch case 語(yǔ)句 | 多條件分支判斷 |
trim,where | 輔助元素 | 用于處理一些SQL拼裝問題 |
foreach | 循環(huán)語(yǔ)句 | 在in語(yǔ)句等列舉條件常用 |
bind | 輔助元素 | 拼接參數(shù) |
3.動(dòng)態(tài) sql 的執(zhí)行原理?
原理為:使用 OGNL 從 sql 參數(shù)對(duì)象中計(jì)算表達(dá)式的值,根據(jù)表達(dá)式的值動(dòng)態(tài)拼接 sql,以此來完成動(dòng)態(tài) sql 的功能。
二、MyBatis標(biāo)簽
1.if標(biāo)簽:條件判斷
MyBatis if 類似于 Java 中的 if 語(yǔ)句,是 MyBatis 中最常用的判斷語(yǔ)句。使用 if 標(biāo)簽可以節(jié)省許多拼接 SQL 的工作,把精力集中在 XML 的維護(hù)上。
1)不使用動(dòng)態(tài)sql
if 語(yǔ)句使用方法簡(jiǎn)單,常常與 test 屬性聯(lián)合使用。語(yǔ)法如下:
SQL語(yǔ)句
2)使用動(dòng)態(tài)sql
上面的查詢語(yǔ)句,我們可以發(fā)現(xiàn),如果 #{username} 為空,那么查詢結(jié)果也是空,如何解決這個(gè)問題呢?使用 if 來判斷,可多個(gè) if 語(yǔ)句同時(shí)使用。
以下語(yǔ)句表示為可以按照網(wǎng)站名稱(name)或者網(wǎng)址(url)進(jìn)行模糊查詢。如果您不輸入名稱或網(wǎng)址,則返回所有的網(wǎng)站記錄。但是,如果你傳遞了任意一個(gè)參數(shù),它就會(huì)返回與給定參數(shù)相匹配的記錄。
2.where+if標(biāo)簽
where、if同時(shí)使用可以進(jìn)行查詢、模糊查詢
注意,
這個(gè)“where”標(biāo)簽會(huì)知道如果它包含的標(biāo)簽中有返回值的話,它就插入一個(gè)‘where’。此外,如果標(biāo)簽返回的內(nèi)容是以AND 或OR 開頭的,則它會(huì)剔除掉。
3.set標(biāo)簽
set可以用來修改
updatestudent wheresid=#{sid} sname=#{sname}, spwd=#{spwd}, sex=#{sex}, phone=#{phone} sid=#{sid}
4.choose(when,otherwise) 語(yǔ)句
有時(shí)候,我們不想用到所有的查詢條件,只想選擇其中的一個(gè),查詢條件有一個(gè)滿足即可,使用 choose 標(biāo)簽可以解決此類問題,類似于 Java 的 switch 語(yǔ)句
也就是說,這里我們有三個(gè)條件,id、username、sex,只能選擇一個(gè)作為查詢條件
如果 id 不為空,那么查詢語(yǔ)句為:select * from user where id=?
如果 id 為空,那么看username 是否為空,如果不為空,那么語(yǔ)句為 select * from user where username=?;
如果 username 為空,那么查詢語(yǔ)句為 select * from user where sex=?
5.trim
trim標(biāo)記是一個(gè)格式化的標(biāo)記,可以完成set或者是where標(biāo)記的功能
①、用 trim 改寫上面第二點(diǎn)的 if+where 語(yǔ)句
prefix:前綴
prefixoverride:去掉第一個(gè)and或者是or
②、用 trim 改寫上面第三點(diǎn)的 if+set 語(yǔ)句
updateuseru whereid=#{id} u.username=#{username}, u.sex=#{sex},
suffix:后綴
suffixoverride:去掉最后一個(gè)逗號(hào)(也可以是其他的標(biāo)記,就像是上面前綴中的and一樣)
③、trim+if同時(shí)使用可以添加
insertintostudent sname, spwd, sex, phone, #{sname}, #{spwd}, #{sex}, #{phone}
6.MyBatis foreach標(biāo)簽
foreach是用來對(duì)集合的遍歷,這個(gè)和Java中的功能很類似。通常處理SQL中的in語(yǔ)句。
foreach 元素的功能非常強(qiáng)大,它允許你指定一個(gè)集合,聲明可以在元素體內(nèi)使用的集合項(xiàng)(item)和索引(index)變量。它也允許你指定開頭與結(jié)尾的字符串以及集合項(xiàng)迭代之間的分隔符。這個(gè)元素也不會(huì)錯(cuò)誤地添加多余的分隔符
你可以將任何可迭代對(duì)象(如 List、Set 等)、Map 對(duì)象或者數(shù)組對(duì)象作為集合參數(shù)傳遞給 foreach。當(dāng)使用可迭代對(duì)象或者數(shù)組時(shí),index 是當(dāng)前迭代的序號(hào),item 的值是本次迭代獲取到的元素。當(dāng)使用 Map 對(duì)象(或者 Map.Entry 對(duì)象的集合)時(shí),index 是鍵,item 是值。
//批量查詢 //批量刪除deletefromstudentwheresidin #{ids}
整合案例
xml
select*fromstudent updatestudent wheresid=#{sid} sname=#{sname}, spwd=#{spwd}, sex=#{sex}, phone=#{phone} sid=#{sid}insertintostudent sname, spwd, sex, phone, #{sname}, #{spwd}, #{sex}, #{phone} deletefromstudentwheresidin #{ids}
測(cè)試類:
packagecom.yzx.test; importcom.yzx.entity.Student; importcom.yzx.mapper.StuMapper; importorg.apache.ibatis.io.Resources; importorg.apache.ibatis.session.SqlSession; importorg.apache.ibatis.session.SqlSessionFactory; importorg.apache.ibatis.session.SqlSessionFactoryBuilder; importorg.junit.After; importorg.junit.Before; importorg.junit.Test; importjava.io.IOException; importjava.io.InputStream; importjava.util.List; publicclassStuTest{ SqlSessionsqlSession=null; InputStreamis=null; @Before publicvoidbefore()throwsIOException{ //1.讀取核心配置文件 is=Resources.getResourceAsStream("sqlMapperConfig.xml"); //2.拿到工廠構(gòu)建類 SqlSessionFactoryBuildersqlSessionFactoryBuilder=newSqlSessionFactoryBuilder(); //3.拿到具體工廠 SqlSessionFactorybuild=sqlSessionFactoryBuilder.build(is); //4.拿到session sqlSession=build.openSession(); } @After publicvoidafter(){ //7,提交事務(wù) sqlSession.commit(); //8.關(guān)閉資源 sqlSession.close(); if(is!=null){ try{ is.close(); }catch(IOExceptione){ e.printStackTrace(); } }; } //查詢所有 @Test publicvoidfind(){ //5.獲取具體的mapper接口 StuMappermapper=sqlSession.getMapper(StuMapper.class); //6.調(diào)用執(zhí)行 Listlist=mapper.find(); list.forEach(a->System.out.println(a)); } //查詢單個(gè) @Test publicvoidfindbyid(){ StuMappermapper=sqlSession.getMapper(StuMapper.class); List list=mapper.findbyid(2); list.forEach(a->System.out.println(a)); } //模糊查詢 @Test publicvoidfindQuery(){ StuMappermapper=sqlSession.getMapper(StuMapper.class); Studentstu=newStudent(); stu.setSname("小"); stu.setSex("男"); List list=mapper.findQuery(stu); list.forEach(a->System.out.println(a)); } //修改 @Test publicvoidupd(){ StuMappermapper=sqlSession.getMapper(StuMapper.class); Studentstu=newStudent(); stu.setSid(3); stu.setSname("小若"); stu.setSex("人妖"); inti=mapper.upd(stu); System.out.println("修改了"+i+"條數(shù)據(jù)"+""+stu.toString()); } //添加 @Test publicvoidadd(){ StuMappermapper=sqlSession.getMapper(StuMapper.class); Studentstu=newStudent(); stu.setSname("小賀"); stu.setSex("男"); stu.setPhone("99999999"); inti=mapper.add(stu); System.out.println("添加了"+i+"條數(shù)據(jù)"+""+stu.toString()); } //批量操作 @Test publicvoidfindAll(){ StuMappermapper=sqlSession.getMapper(StuMapper.class); Integer[]i={1,2,3,4}; List list=mapper.findAll(i); list.forEach(a->System.out.println(a)); } //批量操作 //批量刪除 @Test publicvoiddel(){ StuMappermapper=sqlSession.getMapper(StuMapper.class); Integer[]i={1,2,3,4}; inti1=mapper.del(i); System.out.println("刪除了"+i1+"條數(shù)據(jù)"); } }
7.sql
在實(shí)際開發(fā)中會(huì)遇到許多相同的SQL,比如根據(jù)某個(gè)條件篩選,這個(gè)篩選很多地方都能用到,我們可以將其抽取出來成為一個(gè)公用的部分,這樣修改也方便,一旦出現(xiàn)了錯(cuò)誤,只需要改這一處便能處處生效了,此時(shí)就用到了
當(dāng)多種類型的查詢語(yǔ)句的查詢字段或者查詢條件相同時(shí),可以將其定義為常量,方便調(diào)用。為求
select*fromstudent
8.include
這個(gè)標(biāo)簽和
refid這個(gè)屬性就是指定
9.如何引用其他XML中的SQL片段
比如你在com.xxx.dao.xxMapper這個(gè)Mapper的XML中定義了一個(gè)SQL片段如下:
ID,MAJOR,BIRTHDAY,AGE,NAME,HOBBY
此時(shí)我在com.xxx.dao.PatinetMapper中的XML文件中需要引用,如下:
三、MyBatis關(guān)聯(lián)查詢
1.MyBatis一對(duì)多關(guān)聯(lián)查詢
2.MyBatis多對(duì)一關(guān)聯(lián)查詢
3.MyBatis多對(duì)多關(guān)聯(lián)查詢
審核編輯:劉清
-
SQL
+關(guān)注
關(guān)注
1文章
764瀏覽量
44130 -
JAVA語(yǔ)言
+關(guān)注
關(guān)注
0文章
138瀏覽量
20095 -
XML技術(shù)
+關(guān)注
關(guān)注
0文章
15瀏覽量
6011 -
UTF-8
+關(guān)注
關(guān)注
0文章
13瀏覽量
7854 -
mybatis
+關(guān)注
關(guān)注
0文章
60瀏覽量
6713
原文標(biāo)題:MyBatis 動(dòng)態(tài) SQL 最全教程,這樣寫 SQL 太爽了!
文章出處:【微信號(hào):芋道源碼,微信公眾號(hào):芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論