中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

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

MySQL動態SQL拼接怎么實現

發布時間:2022-12-01 17:35:33 來源:億速云 閱讀:166 作者:iii 欄目:MySQL數據庫

這篇“MySQL動態SQL拼接怎么實現”文章的知識點大部分人都不太理解,所以小編給大家總結了以下內容,內容詳細,步驟清晰,具有一定的借鑒價值,希望大家閱讀完這篇文章能有所收獲,下面我們一起來看看這篇“MySQL動態SQL拼接怎么實現”文章吧。

一、動態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片段,達到重復使用的目的

講解

1. 準備Mybatis環境
  • 創建java項目,導入jar包;準備JavaBean

  • 創建映射器接口UserDao

  • 創建映射配置文件UserDao.xml

  • 創建全局配置文件SqlMapConfig.xml

  • 創建日志配置文件log4j.properties

2. <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();
    }}

3. <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);}

4. <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);
    }

5. <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拼接怎么實現”這篇文章的內容,相信大家都有了一定的了解,希望小編分享的內容對大家有幫助,若想了解更多相關的知識內容,請關注億速云行業資訊頻道。

向AI問一下細節

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

AI

赤城县| 湾仔区| 淮安市| 鹤庆县| 内江市| 南丹县| 定陶县| 旬阳县| 德州市| 清徐县| 布拖县| 唐山市| 葫芦岛市| 澄城县| 鸡西市| 沐川县| 广东省| 玉溪市| 延安市| 汽车| 恭城| 两当县| 广州市| 遂川县| 临江市| 北辰区| 阜城县| 饶阳县| 兴山县| 军事| 章丘市| 东乡| 夏津县| 台东县| 鄱阳县| 施甸县| 运城市| 云梦县| 洛川县| 昌江| 右玉县|