Heim > Datenbank > MySQL-Tutorial > Lassen Sie uns über das dynamische SQL-Spleißen von MySQL sprechen

Lassen Sie uns über das dynamische SQL-Spleißen von MySQL sprechen

WBOY
Freigeben: 2022-12-01 17:20:35
nach vorne
2045 Leute haben es durchsucht

Dieser Artikel vermittelt Ihnen relevantes Wissen über MySQL und stellt hauptsächlich den relevanten Inhalt zum dynamischen SQL-Spleißen vor. In der tatsächlichen Geschäftsentwicklung werden unsere SQL-Anweisungen normalerweise dynamisch gespleißt, z. B. SQL-Anweisungen für die bedingte Suche usw. Nehmen wir eine Schauen Sie sie sich unten an. Ich hoffe, dass sie für alle hilfreich sind.

Lassen Sie uns über das dynamische SQL-Spleißen von MySQL sprechen

Empfohlenes Lernen: MySQL-Video-Tutorial

1. Dynamisches SQL. Splicing

Ziel

  • Be Kann Mybatis-Tags verwenden. Realisieren Sie dynamisches SQL-Spleißen

Analyse

Im vorherigen Lernprozess waren die von uns verwendeten SQL-Anweisungen sehr einfach. In der tatsächlichen Geschäftsentwicklung werden unsere SQL-Anweisungen normalerweise dynamisch gespleißt, beispielsweise SQL-Anweisungen für bedingte Suchfunktionen.

# 提供了一个功能:用户可以在页面上根据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 '%北京%';
Nach dem Login kopieren

In Mybatis werden SQL-Anweisungen in die XML-Datei der Mapping-Konfiguration geschrieben. Mybatis bietet einige XML-Tags zur Implementierung dynamischen SQL-Spleißens.

Häufig verwendete Tags sind:

  • <if></if>: werden zur Beurteilung verwendet, äquivalent zu if-Beurteilung in Java<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>
Nach dem Login kopieren
使用示例
  • 根据用户的名称和性别搜索用户信息。把搜索条件放到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> search1(User user);}
Nach dem Login kopieren
  • 映射文件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="search1" 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>
Nach dem Login kopieren
  • 功能测试,在测试类里加测试方法

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.search1(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();
    }}
Nach dem Login kopieren

3. <where>标签

语法介绍

在刚刚的练习的SQL语句中,我们写了where 1=1。如果不写的话,SQL语句会出现语法错误。Mybatis提供了一种代替where 1=1的技术:<where></where>标签。

代码示例

把上一章节的实现代码进行优化,使用<where></where>标签代替where 1=1

  • 映射器UserDao的search1方法:已有,不用修改

/**
 * 根据username和sex搜索用户
 * @param user 封装了搜索条件的User对象
 * @return 搜索的结果
 */List<User> search1(User user);
Nach dem Login kopieren
  • 在映射文件UserDao.xml里修改SQL语句

<!--
    where标签:让Mybatis帮我们生成一个where关键字
        Mybatis会智能判断:
            如果一个条件都没有,就不生成where关键字
            如果有条件,会判断是否有多余的and关键字,把多余的and去掉
        注意:建议把所有的where条件都放到where标签里边
    --><select id="search1" 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>
Nach dem Login kopieren
  • 在测试类里进行功能测试:测试方法不需要修改

@Testpublic void testSearch(){
    User user = new User();
    // user.setUsername("王");
    // user.setSex("男");

    List<User> userList = userDao.search1(user);
    userList.forEach(System.out::println);}
Nach dem Login kopieren

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);
Nach dem Login kopieren

假如我们传参了id的集合,那么在映射文件中,如何遍历集合拼接SQL语句呢?可以使用foreach标签实现。

<!--
foreach标签:
	属性:
		collection:被循环遍历的对象,使用OGNL表达式获取,注意不要加#{}
		open:循环之前,拼接的SQL语句的开始部分
		item:定义变量名,代表被循环遍历中每个元素,生成的变量名
		separator:分隔符
		close:循环之后,拼接SQL语句的结束部分
	标签体:
		使用#{OGNL}表达式,获取到被循环遍历对象中的每个元素
--><foreach collection="" open="id in(" item="id" separator="," close=")">
    #{id}</foreach>
Nach dem Login kopieren
使用示例
  • 有搜索条件类QueryVO如下:

package com.itheima.domain;public class QueryVO {
    private Integer[] ids;

    public Integer[] getIds() {
        return ids;
    }

    public void setIds(Integer[] ids) {
        this.ids = ids;
    }}
Nach dem Login kopieren
  • 在映射器UserDao里加方法

/**
     * QueryVO里有一个Integer[] ids
     * 要求:根据ids查询对应的用户列表
     */List<User> search2(QueryVO vo);
Nach dem Login kopieren
  • 在映射文件UserDao.xml里配置statement

    <!--
    foreach标签:用于循环遍历
        collection:被循环的集合/数组
        item:定义一个变量
        separator:定义拼接时的分隔符
        open:拼接字符串时的开始部分
        close:拼接字符串时的结束部分

        相当于 for(Integer id: ids){}
        select * from user where id in(41, 42, 45)
    -->
    <select id="search2" 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>
Nach dem Login kopieren
  • 功能测试

    @Test
    public void testSearch2(){
        QueryVO vo = new QueryVO();
        vo.setIds(new Integer[]{41,42,43,44,45});
        List<User> userList = userDao.search2(vo);
        userList.forEach(System.out::println);
    }
Nach dem Login kopieren

5. <sql>标签

在映射文件中,我们发现有很多SQL片段是重复的,比如:select * from user。Mybatis提供了一个<sql>

<where>< /where> ;: Wird normalerweise mit if kombiniert und ersetzt wobei 1=1🎜🎜<foreach></foreach> in SQL-Anweisungen: Wird zum Durchlaufen verwendet eine Sammlung erstellen und den Inhalt der Sammlung in SQL-Anweisungen zusammenfügen. Beispiel: Splicing: in (value1, value2, ...)🎜🎜<sql></sql>: wird verwendet, um SQL-Fragmente für die Wiederverwendung zu definieren 🎜🎜

🎜🎜Erklärung🎜

🎜🎜1 Bereiten Sie die Mybatis-Umgebung vor🎜

    🎜🎜Java-Projekt erstellen, JAR-Paket importieren; JavaBean vorbereiten🎜🎜🎜🎜Mapper-Schnittstelle UserDao erstellen🎜🎜🎜🎜Mapping-Konfigurationsdatei UserDao.xml erstellen🎜🎜🎜🎜Globale Konfigurationsdatei SqlMapConfig .xml erstellen 🎜🎜🎜🎜Erstellen Sie die Protokollkonfigurationsdatei log4j.properties🎜🎜🎜

    🎜🎜2. <if> Tag: 🎜

    🎜Syntax-Einführung🎜
<sql id="唯一标识">sql语句片段</sql>
Nach dem Login kopieren
Nach dem Login kopieren
🎜Verwendungsbeispiele
🎜🎜🎜Suchen Sie Benutzerinformationen basierend auf dem Namen und Geschlecht des Benutzers. Fügen Sie die Suchbedingungen in das User-Objekt ein und übergeben Sie es an die SQL-Anweisung🎜🎜🎜🎜Fügen Sie Methoden auf der Mapper-Schnittstelle UserDao hinzu🎜🎜🎜
<include refid="sql片段的id"></include>
Nach dem Login kopieren
Nach dem Login kopieren
    🎜🎜Mapping-Datei UserDao .xml Anweisung konfigurieren🎜🎜🎜
package com.itheima.domain;/**
 * @author liuyp
 * @date 2021/09/07
 */public class QueryVO {
    private Integer[] ids;
    private User user;

    //get/set方法……}
Nach dem Login kopieren
Nach dem Login kopieren
    🎜🎜Funktionstest, Testmethode in Testklasse hinzufügen🎜🎜🎜
    /**
     * 动态SQL拼接的综合应用:if、where、foreach
     * 要求:QueryVo里有ids、username、sex值,根据这些值进行搜索
     */
    List<User> search3(QueryVO vo);
Nach dem Login kopieren
Nach dem Login kopieren

🎜 🎜 3. <where> tag🎜

🎜🎜Einführung in die Syntax🎜
🎜In der SQL-Anweisung, die wir gerade geübt haben, Wir haben wo 1=1 geschrieben. Wenn Sie es nicht schreiben, weist die SQL-Anweisung Syntaxfehler auf. Mybatis bietet eine Technologie, die das where 1=1: <where></where>-Tag ersetzt. 🎜
🎜Codebeispiel
🎜Optimieren Sie den Implementierungscode im vorherigen Kapitel und verwenden Sie stattdessen das <where></where>-Tag wobei 1=1🎜
    🎜🎜Die Such1-Methode des Mappers UserDao: vorhanden, keine Änderung erforderlich🎜🎜🎜
<select id="search3" 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>
Nach dem Login kopieren
Nach dem Login kopieren
    🎜🎜Ändern Sie die SQL-Anweisung in der Zuordnungsdatei UserDao.xml🎜🎜🎜
    @Test
    public void testSearch3(){
        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.search3(vo);
        userList.forEach(System.out::println);
    }
Nach dem Login kopieren
Nach dem Login kopieren
    🎜🎜Führen Sie Funktionstests im Test durch Klasse: Die Testmethode muss nicht geändert werden id="_226" >🎜Syntax-Einführung🎜foreach-Tag wird normalerweise verwendet, um eine Sammlung zu durchlaufen und den Inhalt der Sammlung in eine SQL-Anweisung zu integrieren. Beispielsweise müssen wir Benutzerinformationen basierend auf mehreren IDs abfragen. Die SQL-Anweisung lautet: 🎜rrreee🎜Wenn wir eine Sammlung von IDs übergeben, wie durchlaufen wir die Sammlung und fügen die SQL-Anweisung in der Zuordnungsdatei zusammen? Dies kann mit dem Tag foreach erreicht werden. 🎜rrreee
    🎜Verwendungsbeispiel
      🎜🎜Es gibt eine Suchbedingungsklasse QueryVO wie folgt: 🎜🎜🎜 rrreee
        🎜🎜Methoden im Mapper UserDao hinzufügen🎜🎜🎜rrreee
          🎜🎜In der Zuordnungsdatei UserDao konfigurieren. xml-Anweisung🎜🎜🎜rrreee
            🎜🎜Funktionstests🎜🎜🎜rrreee

            🎜5. <sql> Tag

            🎜In der Zuordnungsdatei haben wir festgestellt, dass viele SQL-Fragmente wiederholt werden, wie zum Beispiel: select * from user. Mybatis bietet ein <sql>-Tag zum Extrahieren wiederholter SQL-Fragmente und kann wiederverwendet werden. 🎜
            语法介绍

            在映射文件中定义SQL片段:

<sql id="唯一标识">sql语句片段</sql>
Nach dem Login kopieren
Nach dem Login kopieren

在映射文件中引用SQL片段:

<include refid="sql片段的id"></include>
Nach dem Login kopieren
Nach dem Login kopieren
使用示例

在查询用户的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方法……}
Nach dem Login kopieren
Nach dem Login kopieren
  • 在映射器UserDao里加方法

    /**
     * 动态SQL拼接的综合应用:if、where、foreach
     * 要求:QueryVo里有ids、username、sex值,根据这些值进行搜索
     */
    List<User> search3(QueryVO vo);
Nach dem Login kopieren
Nach dem Login kopieren
  • 在映射文件UserDao.xml里配置statement

<select id="search3" 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>
Nach dem Login kopieren
Nach dem Login kopieren
  • 在测试类里加测试方法

    @Test
    public void testSearch3(){
        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.search3(vo);
        userList.forEach(System.out::println);
    }
Nach dem Login kopieren
Nach dem Login kopieren

推荐学习:mysql视频教程

Das obige ist der detaillierte Inhalt vonLassen Sie uns über das dynamische SQL-Spleißen von MySQL sprechen. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Verwandte Etiketten:
Quelle:csdn.net
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage