您好,登錄后才能下訂單哦!
這篇“MySQL動態SQL拼接怎么實現”文章的知識點大部分人都不太理解,所以小編給大家總結了以下內容,內容詳細,步驟清晰,具有一定的借鑒價值,希望大家閱讀完這篇文章能有所收獲,下面我們一起來看看這篇“MySQL動態SQL拼接怎么實現”文章吧。
能夠使用mybatis的標簽實現動態SQL拼接
我們在前邊的學習過程中,使用的SQL語句都非常簡單。而在實際業務開發中,我們的SQL語句通常是動態拼接而成的,比如:條件搜索功能的SQL語句。
# 提供了一個功能:用戶可以在頁面上根據username、sex、address進行搜索
# 用戶輸入的搜索條件:可以是一個條件,也可能是兩個、三個
# 只輸入一個條件:姓名是"王"
SELECT * FROM USER WHERE username LIKE '%王%'
# 只輸入一個條件:性別是“男”
SELECT * FROM USER WHERE sex = '男'
# 輸入兩個條件:姓名“王”,性別“男”
SELECT * FROM USER WHERE username LIKE '%王%' AND sex = '男'
# 輸入三個條件:姓名“王”,性別“男”,地址“北京”
SELECT * FROM USER WHERE username LIKE '%王%' AND sex = '男' AND address LIKE '%北京%';
在Mybatis中,SQL語句是寫在映射配置的XML文件中的。Mybatis提供了一些XML的標簽,用來實現動態SQL的拼接。
常用的標簽有:
<if></if>
:用來進行判斷,相當于Java里的if判斷
<where></where>
:通常和if配合,用來代替SQL語句中的where 1=1
<foreach></foreach>
:用來遍歷一個集合,把集合里的內容拼接到SQL語句中。例如拼接:in (value1, value2, ...)
<sql></sql>
:用于定義sql片段,達到重復使用的目的
創建java項目,導入jar包;準備JavaBean
創建映射器接口UserDao
創建映射配置文件UserDao.xml
創建全局配置文件SqlMapConfig.xml
創建日志配置文件log4j.properties
<if>
標簽:<if test="判斷條件,使用OGNL表達式進行判斷">
SQL語句內容, 如果判斷為true,這里的SQL語句就會進行拼接</if>
根據用戶的名稱和性別搜索用戶信息。把搜索條件放到User對象里,傳遞給SQL語句
映射器接口UserDao上加方法
package com.demo.dao;import com.demo.domain.User;import java.util.List;public interface UserDao {
/**
* 根據username和sex搜索用戶
* @param user 封裝了搜索條件的User對象
* @return 搜索的結果
*/
List<User> search2(User user);}
映射文件UserDao.xml里配置statement
<?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd"><mapper namespace="com.itheima.dao.UserDao">
<!--
if標簽:用于條件判斷
語法:<if test="用OGNL表達式判斷"> 如果判斷為true,這里的內容會拼接上去 </if>
注意:標簽里寫OGNL表達式,不要再加#{}、${}
常用的OGNL表達式:
比較:>, <, >=, <=, ==, != 或者 gt, lt, gte, lte, eq, neq
邏輯:&&,||,! 或者 and, or, not
調用方法:username.length(), list.size()
-->
<select id="search2" resultType="User">
select * from user where 1=1 <if test="username != null and username.length()>0">
and username like "%"#{username}"%" </if>
<if test="sex != null and sex.length()>0">
and sex = #{sex} </if>
</select></mapper>
功能測試,在測試類里加測試方法
package com.demo;import com.demo.dao.UserDao;import com.demo.domain.User;import org.apache.ibatis.io.Resources;import org.apache.ibatis.session.SqlSession;import org.apache.ibatis.session.SqlSessionFactory;import org.apache.ibatis.session.SqlSessionFactoryBuilder;import org.junit.After;import org.junit.Before;import org.junit.Test;import java.io.IOException;import java.io.InputStream;import java.util.List;public class SqlTest {
private UserDao userDao;
private SqlSession session;
private InputStream is;
/**
* 要求:根據username和sex搜索用戶
* 搜索條件放到user對象里
*/
@Test
public void testSearch(){
User user = new User();
// user.setUsername("王");
// user.setSex("男");
List<User> userList = userDao.search2(user);
userList.forEach(System.out::println);
}
@Before
public void init() throws IOException {
//1. 讀取全局配置文件
is = Resources.getResourceAsStream("SqlMapConfig.xml");
//2. 得到一個SqlSession對象
SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(is);
session = factory.openSession();
userDao = session.getMapper(UserDao.class);
}
@After
public void destroy() throws IOException {
session.close();
is.close();
}}
<where>
標簽在剛剛的練習的SQL語句中,我們寫了where 1=1
。如果不寫的話,SQL語句會出現語法錯誤。Mybatis提供了一種代替where 1=1
的技術:<where></where>
標簽。
把上一章節的實現代碼進行優化,使用<where></where>
標簽代替where 1=1
映射器UserDao的search2方法:已有,不用修改
/**
* 根據username和sex搜索用戶
* @param user 封裝了搜索條件的User對象
* @return 搜索的結果
*/List<User> search2(User user);
在映射文件UserDao.xml里修改SQL語句
<!--
where標簽:讓Mybatis幫我們生成一個where關鍵字
Mybatis會智能判斷:
如果一個條件都沒有,就不生成where關鍵字
如果有條件,會判斷是否有多余的and關鍵字,把多余的and去掉
注意:建議把所有的where條件都放到where標簽里邊
--><select id="search2" resultType="User">
select * from user <where>
<if test="username != null and username.length()>0">
and username like "%"#{username}"%" </if>
<if test="sex != null and sex.length()>0">
and sex = #{sex} </if>
</where></select>
在測試類里進行功能測試:測試方法不需要修改
@Testpublic void testSearch(){
User user = new User();
// user.setUsername("王");
// user.setSex("男");
List<User> userList = userDao.search2(user);
userList.forEach(System.out::println);}
<foreach>
標簽foreach標簽,通常用于循環遍歷一個集合,把集合的內容拼接到SQL語句中。例如,我們要根據多個id查詢用戶信息,SQL語句:
select * from user where id = 1 or id = 2 or id = 3;select * from user where id in (1, 2, 3);
假如我們傳參了id的集合,那么在映射文件中,如何遍歷集合拼接SQL語句呢?可以使用foreach
標簽實現。
<!--
foreach標簽:
屬性:
collection:被循環遍歷的對象,使用OGNL表達式獲取,注意不要加#{}
open:循環之前,拼接的SQL語句的開始部分
item:定義變量名,代表被循環遍歷中每個元素,生成的變量名
separator:分隔符
close:循環之后,拼接SQL語句的結束部分
標簽體:
使用#{OGNL}表達式,獲取到被循環遍歷對象中的每個元素
--><foreach collection="" open="id in(" item="id" separator="," close=")">
#{id}</foreach>
有搜索條件類QueryVO如下:
package com.itheima.domain;public class QueryVO {
private Integer[] ids;
public Integer[] getIds() {
return ids;
}
public void setIds(Integer[] ids) {
this.ids = ids;
}}
在映射器UserDao里加方法
/**
* QueryVO里有一個Integer[] ids
* 要求:根據ids查詢對應的用戶列表
*/List<User> search3(QueryVO vo);
在映射文件UserDao.xml里配置statement
<!--
foreach標簽:用于循環遍歷
collection:被循環的集合/數組
item:定義一個變量
separator:定義拼接時的分隔符
open:拼接字符串時的開始部分
close:拼接字符串時的結束部分
相當于 for(Integer id: ids){}
select * from user where id in(41, 42, 45)
-->
<select id="search3" resultType="User">
<!--select * from user where id in(41, 42, 45)-->
select * from user where <foreach collection="ids" open="id in(" item="id" separator="," close=")">
#{id} </foreach>
</select>
功能測試
@Test
public void testSearch3(){
QueryVO vo = new QueryVO();
vo.setIds(new Integer[]{41,42,43,44,45});
List<User> userList = userDao.search3(vo);
userList.forEach(System.out::println);
}
<sql>
標簽在映射文件中,我們發現有很多SQL片段是重復的,比如:select * from user
。Mybatis提供了一個<sql>
標簽,把重復的SQL片段抽取出來,可以重復使用。
在映射文件中定義SQL片段:
<sql id="唯一標識">sql語句片段</sql>
在映射文件中引用SQL片段:
<include refid="sql片段的id"></include>
在查詢用戶的SQL中,需要重復編寫:select * from user
。把這部分SQL提取成SQL片段以重復使用
要求:QueryVO里有ids,user對象。根據條件進行搜索
修改QueryVO,增加成員變量user
package com.itheima.domain;/**
* @author liuyp
* @date 2021/09/07
*/public class QueryVO {
private Integer[] ids;
private User user;
//get/set方法……}
在映射器UserDao里加方法
/**
* 動態SQL拼接的綜合應用:if、where、foreach
* 要求:QueryVo里有ids、username、sex值,根據這些值進行搜索
*/
List<User> search4(QueryVO vo);
在映射文件UserDao.xml里配置statement
<select id="search4" resultType="User">
<!--select * from user-->
<include refid="selUser"/>
<where>
<if test="ids != null and ids.length > 0">
<foreach collection="ids" open="and id in(" item="id" separator="," close=")">
#{id} </foreach>
</if>
<!--<if test="user != null">
<if test="user.username != null and user.username.length() > 0">
and username like "%"#{user.username}"%"
</if>
<if test="user.sex != null and user.sex.length() > 0">
and sex = #{user.sex}
</if>
</if>-->
<include refid="userCondition"/>
</where></select><!--
sql標簽:用于定義一個sql片段
include標簽:什么時候要引用某個SQL片段,就使用include標簽
注意:引入SQL片段之后,最終的SQL語句必須要完全符合語法
--><sql id="selUser">select * from user</sql><sql id="userCondition">
<if test="user != null">
<if test="user.username != null and user.username.length() > 0">
and username like "%"#{user.username}"%" </if>
<if test="user.sex != null and user.sex.length() > 0">
and sex = #{user.sex} </if>
</if></sql>
在測試類里加測試方法
@Test
public void testSearch4(){
QueryVO vo = new QueryVO();
vo.setIds(new Integer[]{41,42,43,44,45});
// User user = new User();
// user.setUsername("王");
// user.setSex("男");
// vo.setUser(user);
List<User> userList = userDao.search4(vo);
userList.forEach(System.out::println);
}
以上就是關于“MySQL動態SQL拼接怎么實現”這篇文章的內容,相信大家都有了一定的了解,希望小編分享的內容對大家有幫助,若想了解更多相關的知識內容,請關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。