MySQL动态SQL拼接实例详解
作者:一切如你i 发布时间:2024-01-20 15:13:55
目标
能够使用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> search1(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="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>
功能测试,在测试类里加测试方法
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();
}
}
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);
在映射文件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>
在测试类里进行功能测试:测试方法不需要修改
@Test
public void testSearch(){
User user = new User();
// user.setUsername("王");
// user.setSex("男");
List<User> userList = userDao.search1(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> search2(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="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>
功能测试
@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);
}
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> search3(QueryVO vo);
在映射文件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>
在测试类里加测试方法
@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);
}
来源:https://blog.csdn.net/yiqieruni/article/details/127617481
猜你喜欢
- python修改大数据文件时,如果全加载到内存中,可能会导致内存溢出。因此可借用如下方法,将分件分段读取修改。with open('
- 终于构建出了第一个神经网络,Keras真的很方便。之前不知道Keras这么方便,在构建神经网络的过程中绕了很多弯路,最开始学的TensorF
- 第一步:建与mysql同构的oracle数据库实例,并更新数据。 1.期望将my
- 原则一:注意WHERE子句中的连接顺序: ORACLE采用自下而上的顺序解析WHERE子句,根据这个原理,表之间的连接必须写在其他WHERE
- 如果使用默认的数据库 SQLite3,则无需配置settings.py使用其他数据库,则需要配置settings.py,这里以Mysql为例
- 代码'########### '检测远程文件是否存在 '########### function CheckURL(
- 【摘要】用定值 Cookie 实现反爬 Cookie 生成由于本案例需要用到一个特定的 Cookie ,所以我们需要提前将其生成,你可以直接
- 使用shell调用一个python文件,并向shell中传入参数,举例如下:p1='wang'p2='shuang&
- queue模块简介queue模块是Python内置的标准模块,模块实现了三种类型的队列,它们的区别仅仅是条目取回的顺序,分别由3个类进行表示
- 目录GitHub 消息的问题解决方案代码实现0.环境准备1、模拟登录github2.模拟进入Inbox3.检查僵尸项目4.取消关注僵尸项目5
- 导读:这篇论坛文章主要介绍了使用SQL Server升级顾问的具体步骤,详细内容请参考下文。微软提供了SQL Server 2008升级顾问
- 本文介绍了Python字符串格式化,主要有两种方法,分享给大家,具体如下用于字符串的拼接,性能更优。字符串格式化有两种方式:百分号方式、fo
- 在编程过程中,多了解语言周边的一些知识,以及一些技巧,可以让你加速成为一个优秀的程序员。对于Python程序员,你需要注意一下本文所提到的这
- 本文介绍如何利用带进度条的ASP无组件实现断点续传下载大文件。<%@LANGUAGE="VBSCRIPT"&nbs
- 现在网页的设计都讲究整体统一风格,无论是网页的文字、图像,还是浏览器的滚动条都要求颜色和风
- 1.函数调用# 1.调用函数,需要知道函数的名称和参数# 2.调用函数传入的参数需要和函数定义的参数数量和类型一致# 如调用abs函数pri
- 排序问题最近看了极客时间上 《MySQL实战45讲》,纠正了一直以来对 InnoDB 二级索引的一个理解不到位,正好把相关内容总结下。PS:
- 前言数据分析就是将数据以各种图表的形式展现给领导,供领导做决策用,因此熟练掌握饼图、柱状图、线图等图表制作是一个数据分析师必备的技能。Pyt
- 1.1 什么是Mysql多实例?简单的说,Mysql多实例就是在一台服务器上同时开启多个不同的服务端口(如 : 3306/3307/3308
- (可能只有最后一句命令有用,可能全篇都没用)(小白方法,可能只适用于本人情况)安装matplotlib时,出现的三种失败情况1、read t