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