MySQL动态SQL拼接怎么实现

网友投稿 849 2023-11-19

MySQL动态SQL拼接怎么实现

这篇“MySQL动态SQL拼接怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“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上加方法

packagecom.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;importorg.apache.ibatis.io.Resources;import org.apache.ibatis.session.SqlSession;importorg.apache.ibatis.session.SqlSessionFactory;import org.apache.ibatis.session.SqlSessionFactoryBuilder;import org.junit.After;importorg.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 = newSqlSessionFactoryBuilder().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搜索用户  * @paramuser 封装了搜索条件的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 (123);

假如我们传参了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;/**  * @authorliuyp  *@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拼接怎么实现”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注行业资讯频道。

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:如何下载安装MySQL
下一篇:弹性扩缩容使用方法:优化你的资源管理策略
相关文章