MyBatis 的强大特性之一便是它的动态 SQL。如果你有使用 JDBC 或其他类似框架的经验,你就能体会到根据不同条件拼接 SQL 语句有多么痛苦。拼接的时候要确保不能忘了必要的空格,还要注意省掉列名列表最后的逗号。利用动态 SQL 这
前言
前面mysql都是通过静态sql进行查询的,但是如果业务复杂的时候,我们会遇到引号问题,或者多一个空格,这就使得sql代码编写错误了,所以为了解决这个问题,我们有了动态sql。
mybatis框架的动态sql技术是一种根据特定条件动态拼装sql语句的功能,它存在的意义是为了解决拼接sql语句字符串时的痛点问题。具体是通过标签来实现的。
动态sql
1.先看一下模块目录结构
在类路径的resources下的mapper包下创建sql.xml文件(共性抽取)
2.物理建模和逻辑建模
这里省略物理建模步骤,要求数据库的表与pojo类要对应。package pojo;
import lombok.allargsconstructor;
import lombok.data;
import lombok.noargsconstructor;
@data
@allargsconstructor
@noargsconstructor
public class employee {
private integer empid;
private string empname;
private double empsalary;
}
3. 引入依赖
把之前的log4j复制到类路径resouces下,另外我们引入依赖后的pom.xml如下:<?xml version="1.0" encoding="utf-8"?>
<project xmlns="http://maven.apache.org/pom/4.0.0"
xmlns:xsi="http://www.w3.org/2001/xmlschema-instance"
xsi:schemalocation="http://maven.apache.org/pom/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelversion>4.0.0</modelversion>
<groupid>org.example</groupid>
<artifactid>day03-mybatis02-dynamic</artifactid>
<version>1.0-snapshot</version>
<packaging>jar</packaging>
<dependencies>
<dependency>
<groupid>org.projectlombok</groupid>
<artifactid>lombok</artifactid>
<version>1.18.8</version>
<scope>provided</scope>
</dependency>
<!-- mybatis核心 -->
<dependency>
<groupid>org.mybatis</groupid>
<artifactid>mybatis</artifactid>
<version>3.5.7</version>
</dependency>
<!-- junit测试 -->
<dependency>
<groupid>junit</groupid>
<artifactid>junit</artifactid>
<version>4.12</version>
<scope>test</scope>
</dependency>
<!-- mysql驱动 -->
<dependency>
<groupid>mysql</groupid>
<artifactid>mysql-connector-java</artifactid>
<version>5.1.3</version>
<scope>runtime</scope>
</dependency>
<!-- log4j日志 -->
<dependency>
<groupid>log4j</groupid>
<artifactid>log4j</artifactid>
<version>1.2.17</version>
</dependency>
</dependencies>
</project>
4.全局配置文件 <?xml version="1.0" encoding="utf-8" ?>
<!doctype configuration
public "-//mybatis.org//dtd config 3.0//en"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<!--驼峰映射-->
<settings>
<setting name="mapunderscoretocamelcase" value="true"/>
</settings>
<!--类型别名映射-->
<typealiases>
<package name="pojo"/>
</typealiases>
<!--环境配置-->
<environments default="dev">
<environment id="dev">
<transactionmanager type="jdbc"></transactionmanager>
<datasource type="pooled">
<property name="username" value="root"/>
<property name="password" value="888888"/>
<property name="url" value="jdbc:mysql://localhost:3306/mybatis-example"/>
<property name="driver" value="com.mysql.jdbc.driver"/>
</datasource>
</environment>
</environments>
<!--路径映射-->
<mappers>
<mapper resource="mapper/sql.xml"/>
<package name="mapper"/>
</mappers>
</configuration>
注意: 这里有驼峰映射,别名映射,路径映射和路径映射。和以前的不同的是,我们这里做了sql语句的共性抽取,所以得加一个sql的路径映射 <mapper resource="mapper/sql.xml"/>。
5.sql共性抽取文件
在类路径resources下的包mapper下创建一个sql.xml(因为我们sql是要写在映射文件中,自己本身也是映射文件,所以需要写在mapper下)。到要用的时候,在映射路径文件中需要用到这个sql语句的地方加入 <include refid="mapper.sql.myselectsql"></include>。<?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="mapper.sql">
<sql id="myselectsql">
select emp_id,emp_name,emp_salary from t_emp
</sql>
</mapper>
共性抽取文件也可以不配置,这时候直接在映射文件中把要执行的语句重新编写就行了。
6.mapper接口
一共有七个方法package mapper;
import org.apache.ibatis.annotations.param;
import pojo.employee;
import java.util.list;
public interface employeemapper {
//根据员工的empid查询大于该empid的所有员工,如果empid为null,则查询全体员工
list<employee> selectemployeelistbyempid(integer empid);
/**
* 查询大于传入的empid并且工资大于传入的empsalary的员工集合,如果传入的empid为null,则不考虑empid条件
* 传入的empsalary为null则不考虑empsalary的条件
*/
list<employee> selectemployeelistbyempidandempsalary(@param("empid") integer empid, @param("empsalary") double empsalary);
/**
* 根据empid更新员工信息,如果某个值为null,则不更新这个字段
*/
void updateemployee(employee employee);
/**
* 根据emp_id查询员工信息,如果0<emp_id<6,那么就查询所有大于该emp_id的员工,如果emp_id是大于6,那么就查询所有小于该emp_id的员工
* 如果是其它情况,则查询所有员工信息
*/
list<employee> selectemployeelist(integer empid);
/**
* 添加员工信息
*/
void insertemployee(employee employee);
/**
* 批量添加员工集合
*/
void insertemployeelist(@param("employeelist") list<employee> employeelist);
/**
* 根据员工的id集合查询员工集
*/
list<employee> selectemployeelistbyempidlist(list<integer> idlist);
}
if
目标:根据员工的empid查询大于该empid的所有员工,如果empid为null,则查询全体员工。
dao接口的方法为:
list<employee> selectemployeelistbyempid(integer empid);
静态sql: <select id="selectemployeelistbyempid" resulttype="employee">
<include refid="mapper.sql.myselectsql"></include> where emp_id>#{empid}
</select>
动态sql: <select id="selectemployeelistbyempid" resulttype="employee">
<include refid="mapper.sql.myselectsql"></include>
<if test="empid != null">
where emp_id>#{empid}
</if>
</select>
<include refid="mapper.sql.myselectsql"></include>表示引用抽取出的sql片段,也可以直接写sql语句。如果是静态sql,当id为null时,查询出来的是空,动态sql则可以查出全部。if标签里面有test属性名,作为判断语句。
where
目标:
查询大于传入的empid并且工资大于传入的empsalary的员工集合
如果传入的empid为null,则不考虑empid条件
传入的empsalary为null则不考虑empsalary的条件
dao接口方法:
list<employee> selectemployeelistbyempidandempsalary(@param("empid") integer empid, @param("empsalary") double empsalary);
用if标签的动态sql: <select id="selectemployeelistbyempidandempsalary" resulttype="employee">
<include refid="mapper.sql.myselectsql"></include> where
<if test="empid != null">
emp_id>#{empid}
</if>
<if test="empsalary != null">
and emp_salary>#{empsalary}
</if>
这里可以看到,如果empsalary为空,那么sql语句为select * from t_emp where emp_id >#{empid},但是如果empid为空,那么sql语句为select * from t_emp where and emp_salary>#{empsalary},很明显这个是错的,if标签在这里就不适用了。所以我们用where标签,或者trim标签。
where和if的动态sql: <select id="selectemployeelistbyempidandempsalary" resulttype="employee">
<include refid="mapper.sql.myselectsql"></include>
<where>
<if test="empid != null">
emp_id>#{empid}
</if>
<if test="empsalary != null">
and emp_salary>#{empsalary}
</if>
</where>
</select>
where标签的作用:
在第一个条件之前自动添加where关键字
自动去掉第一个条件前的连接符(and、or等等)
trim
trim是修建的意思,其实就是去头去尾,这里还是根据上面那个方法
trim的动态sql <select id="selectemployeelistbyempidandempsalary" resulttype="employee">
<include refid="mapper.sql.myselectsql"></include>
<trim prefix="where" prefixoverrides="and|or">
<if test="empid != null">
emp_id>#{empid}
</if>
<if test="empsalary != null">
and emp_salary>#{empsalary}
</if>
</trim>
</select>
trim标签:
prefix:指定要动态添加的前缀
suffix属性:指定要动态添加的后缀
prefixoverrides:指定要动态去掉的前缀,使用“|”分隔有可能的多个值
suffixoverrides属性:指定要动态去掉的后缀,使用“|”分隔有可能的多个值
set
目标:根据empid更新员工信息,如果某个值为null,则不更新这个字段
dao接口方法:
void updateemployee(employee employee);
我们先用上面的trim标签来解决一下这个问题,
trim的动态sql: <update id="updateemployee" >
<trim prefix="set" prefixoverrides=",">
<if test="empname!=null">
emp_name=#{empname}
</if>
<if test="empsalary!=null">
, emp_salary=#{empsalary}
</if>
</trim>
where emp_id=#{empid}
</update>
set的动态sql <update id="updateemployee" >
update t_emp
<set >
<if test="empname!=null">
emp_name=#{empname}
</if>
<if test="empsalary!=null">
, emp_salary=#{empsalary}
</if>
</set>
可以看出
set标签的作用:
自动在要修改的第一个字段之前添加set关键字
去掉要修改的第一个字段前的连接符(,)
choose、when、otherwise
目标:
根据emp_id查询员工信息,如果0<emp_id<6,那么就查询所有大于该emp_id的员工
如果emp_id是大于6,那么就查询所有小于该emp_id的员工
如果是其它情况,则查询所有员工信息
dao接口方法:
list<employee> selectemployeelist(integer empid);
动态sql <select id="selectemployeelist" resulttype="employee">
<include refid="mapper.sql.myselectsql"></include> where
<choose>
<!--<是<号的转义字符-->
<when test="empid>0 and empid<6">
emp_id>#{empid}
</when>
<when test="empid>6">
emp_id<#{empid}
</when>
<otherwise>
1==1
</otherwise>
</choose>
</select>
choose、when、otherwise
相当于if ... else if... else if ... else
如果某一个when的条件成立,则不会继续判断后续的when
如果所有的when都不成立,则会拼接otherwise标签中的内容
foreach
目标1:批量添加员工信息
dao接口方法:
void insertemployeelist(@param("employeelist") list employeelist);
1.动态sql <insert id="insertemployeelist">
insert into t_emp(emp_name,emp_salary)values
<!--collection标签可以写list,collection,
或者自己自己定义参数名@param("employeelist") list<employee> employeelist-->
<foreach collection="employeelist" separator="," item="emp">
(#{emp.empname},#{emp.empsalary})
</foreach>
</insert>
目标2:根据多个id查询多个员工信息
dao接口
list selectemployeelistbyempidlist(list idlist);
2.动态sql <select id="selectemployeelistbyempidlist" resulttype="employee">
<include refid="mapper.sql.myselectsql"></include>
<foreach collection="collection" item="id" separator="," open="where emp_id in (" close=")">
#{id}
</foreach>
</select>
批量查询:foreach标签
collection属性: 表示要遍历的对象,如果要遍历的参数使用@param注解取名了就使用该名字,如果没有取名list,或者collection。
item属性: 表示遍历出来的元素,我们到时候要拼接sql语句就得使用这个元素: 如果遍历出来的元素是pojo对象, 那么我们就通过 #{遍历出来的元素.pojo的属性} 获取数据;如果遍历出来的元素是简单类型的数据,那么我们就使用 #{遍历出来的元素} 获取这个简单类型数据
separator属性: 遍历出来的元素之间的分隔符
open属性: 在遍历出来的第一个元素之前添加前缀
close属性: 在遍历出来的最后一个元素之后添加后缀
测试程序 import mapper.employeemapper;
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 pojo.employee;
import java.io.inputstream;
import java.util.arraylist;
import java.util.list;
public class test {
private employeemapper employeemapper;
private inputstream is;
private sqlsession sqlsession;
@before
public void init() throws exception{
//目标:获取employeemapper接口的代理对象,并且使用该对象调用selectemployee(1)方法,然后返回employee对象
//1. 将全局配置文件转成字节输入流
is = resources.getresourceasstream("mybatisconfig.xml");
//2. 创建sqlsessionfactorybuilder对象
sqlsessionfactorybuilder sqlsessionfactorybuilder = new sqlsessionfactorybuilder();
//3. 使用构建者模式创建sqlsessionfactory对象
sqlsessionfactory sqlsessionfactory = sqlsessionfactorybuilder.build(is);
//4. 使用工厂模式创建一个sqlsession对象
sqlsession = sqlsessionfactory.opensession();
//5. 使用动态代理模式,创建employeemapper接口的代理对象
employeemapper = sqlsession.getmapper(employeemapper.class);
}
@after
public void after() throws exception{
//提交事务!!!
sqlsession.commit();
//7. 关闭资源
is.close();
sqlsession.close();
}
@org.junit.test
public void testselectemployeelistbyempid(){
system.out.println(employeemapper.selectemployeelistbyempid(null));
}
@org.junit.test
public void testselectemployeelistbyempidandempsalary(){
system.out.println(employeemapper.selectemployeelistbyempidandempsalary(2, 300d));
}
@org.junit.test
public void testupdateemployee(){
employee employee = new employee(3,"celia", 9000d);
employeemapper.updateemployee(employee);
}
@org.junit.test
public void testselectemployeelist(){
system.out.println(employeemapper.selectemployeelist(7));
}
@org.junit.test
public void testinsertemployee(){
employeemapper.insertemployee(new employee(null,"tom",300d));
}
@org.junit.test
public void testinsertemployeelist(){
list<employee> employeelist = new arraylist<>();
for (int i = 11; i <=20 ; i++) {
employeelist.add(new employee(null,"aobama"+i,2000d));
}
employeemapper.insertemployeelist(employeelist);
}
@org.junit.test
public void testselectemployeelistbyempidlist(){
list<integer> idlist = new arraylist<>();
idlist.add(23);
idlist.add(33);
idlist.add(32);
idlist.add(21);
idlist.add(22);
system.out.println(employeemapper.selectemployeelistbyempidlist(idlist));
}
}
到此这篇关于mybatis 动态sql全面详解的文章就介绍到这了,更多相关mybatis 动态sql内容请搜索CodeAE代码之家 以前的文章或继续浏览下面的相关文章希望大家以后多多支持CodeAE代码之家!
原文链接:https://www.cnblogs.com/jasmine-e/p/15354425.html