这篇“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片段,达到重复使用的目的
讲解 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 test="user != null"> <if test="user.username != null and user.username.length() > 0"> and username like "%"#{user.username}"%" <if test="user.sex != null and user.sex.length() > 0"> and sex = #{user.sex} </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拼接怎么实现”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注行业资讯频道。