动态SQL

1.if

  假设需要对Product执行两条sql语句,一个是查询所有,一个是根据名称模糊查询。那么按照现在的方式,必须提供两条sql语句:listProduct和listProductByName然后在调用的时候,分别调用它们来执行。如下所示:

Product.xml

 <?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="mybatis.pojo"> <select id="listProduct1" resultType="Product">
select * from product
</select>
<select id="listProductByName" resultType="Product">
select *
from product where name like concat('%',#{name},'%')
</select>
</mapper>

Test.java

 package mybatis.test;

 import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.List;
import java.util.Map; import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder; import mybatis.pojo.Product; public class testIf {
public static void main(String[] args) throws IOException {
String resource = "mybatis-config.xml";
InputStream inputStream = org.apache.ibatis.io.Resources.getResourceAsStream(resource);
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
SqlSession session = sqlSessionFactory.openSession(); System.out.println("查询所有的");
List<Product> products = session.selectList("listProduct1");
for (Product product : products) {
System.out.println(product);
} System.out.println("模糊查询");
Map<String, Object> params = new HashMap<String, Object>();
params.put("name", "a");
List<Product> products2 = session.selectList("listProductByName",
params);
for (Product product : products2) {
System.out.println(product);
} session.commit();
session.close();
}
}

  如果Product的字段比较多的话,为了应付各个字段的查询,那么就需要写多条sql语句,这样就变得难以维护。这个时候,就可以使用Mybatis 动态SQL里的if标签

 <select id="listProduct2" resultType="Product">
select * from product
<if test="name!=null">
where name like concat('%',#{name},'%')
</if>
</select>

  如果没有传参数name,那么就查询所有,如果有name参数,那么就进行模糊查询。这样只需要定义一条sql语句即可应付多种情况了,在测试的时候,也只需要调用这么一条sql语句listProduct 即可。

         System.out.println("查询所有的");
List<Product> products = session.selectList("listProduct2");
for (Product product : products) {
System.out.println(product);
} System.out.println("模糊查询");
Map<String, Object> params = new HashMap<String, Object>();
params.put("name", "a");
List<Product> products2 = session.selectList("listProduct2", params);
for (Product product : products2) {
System.out.println(product);
}

2.where

<1>如果要进行多条件判断,如下所示:

     <select id="listProduct2" resultType="Product">
select * from product
<if test="name!=null">
where name like concat('%',#{name},'%')
</if>
<if test="price!=0">
and price>=#{price}
</if>
</select>
         System.out.println("多条件查询");
Map<String, Object> params = new HashMap<String, Object>();
params.put("name", "a");
params.put("price", "10");
List<Product> products2 = session.selectList("listProduct4", params);
for (Product product : products2) {
System.out.println(product);
}

这么写的问题是:当没有name参数,却有price参数的时候,执行的sql语句就会是:select * from product_ and price > 10。这样就会报错。

这个问题可以通过<where>标签来解决,如代码所示:

     <select id="listProduct4" resultType="Product">
select * from product
<where>
<if test="name!=null">
and name like concat('%',#{name},'%')
</if>
<if test="price!=null and price!=0">
and price>#{price}
</if>
</where>
</select>

<where>标签会进行自动判断:
如果任何条件都不成立,那么就在sql语句里就不会出现where关键字
如果有任何条件成立,会自动去掉多出来的 and 或者 or。

所以map里面两个参数无论是否提供值都可以正常执行。

<2>与where标签类似的,在update语句里也会碰到多个字段相关的问题。 在这种情况下,就可以使用set标签:

     <update id="updateProduct" parameterType="Product">
update product
<set>
<if test="name!=null">name=#{name},</if>
<if test="price!=null">price=#{price}</if>
</set>
where id=#{id}
</update>
         Product product = new Product();
product.setId(6);
product.setName("product xzz");
product.setPrice(6.99f);
session.update("updateProduct", product);

<3>trim 用来定制想要的功能,比如where标签就可以用以下代码替换:

 <trim prefix="WHERE" prefixOverrides="AND |OR ">
...
</trim>

set标签可以用以下代码来替换:

 <trim prefix="SET" suffixOverrides=",">
...
</trim>

示例:

     <select id="listProduct5" resultType="Product">
select * from product
<trim prefix="WHERE" prefixOverrides="AND|OR">
<if test="name!=null">
and name like concat('%',#{name},'%')
</if>
<if test="price!=null and price!=0">
and price>#{price}
</if>
</trim>
</select>
     <update id="updateProduct2" parameterType="Product">
update product
<trim prefix="SET" suffixOverrides=",">
<if test="name!=null">name=#{name},</if>
<if test="price!=null">price=#{price}</if>
</trim>
where id=#{id}
</update>

3.choose

Mybatis里面没有else标签,但是可以使用when otherwise标签来达到这样的效果。

     <select id="listProduct6" resultType="Product">
select * from product
<where>
<choose>
<when test="name!=null">
and name like concat('%',#{name},'%')
</when>
<when test="price!=null and price!=0">
and price>#{price}
</when>
<otherwise>
and id>1
</otherwise>
</choose>
</where>
</select>

作用: 提供了任何条件,就进行条件查询,否则就使用id>1这个条件。

         System.out.println("多条件查询");
Map<String, Object> params = new HashMap<String, Object>();
// params.put("name", "a");
// params.put("price", "10");
List<Product> products2 = session.selectList("listProduct6", params);
for (Product products : products2) {
System.out.println(products);
}

查询结果:

如果去掉注释,提供查询条件,则结果如下:

4.foreach

适用情况如下所示:

     <select id="listProduct7" resultType="Product">
select * from product
where id in
<foreach item="item" index="index" collection="list" open="("
separator="," close=")">
#{item}
</foreach>
</select>

collection :collection属性的值有三个分别是list、array、map三种,分别对应的参数类型为:List、数组、map集合

item : 表示在迭代过程中每一个元素的别名     

index :表示在迭代过程中每次迭代到的位置(下标)     

open :前缀     

close :后缀     

separator :分隔符,表示迭代时每个元素之间以什么分隔

测试:

         List<Integer> idsIntegers = new ArrayList<Integer>();
idsIntegers.add(1);
idsIntegers.add(3);
idsIntegers.add(5); List<Product> products = session.selectList("listProduct7", idsIntegers); for (Product product : products) {
System.out.println(product);
}

结果:

5.bind

bind标签就像是再做一次字符串拼接,方便后续使用。如下所示,在模糊查询的基础上,把模糊查询改为bind标签。

     <select id="listProductByName" resultType="Product">
select *
from product
where name like concat('%',#{name},'%')
</select>
     <select id="listProductByName2" resultType="Product">
<bind name="likename" value="'%'+name+'%'"/>
select *
from product
where name like #{likename}
</select>

笔记53 Mybatis快速入门(四)的更多相关文章

  1. MyBatis学习笔记(一)——MyBatis快速入门

    转自孤傲苍狼的博客:http://www.cnblogs.com/xdp-gacl/p/4261895.html 一.Mybatis介绍 MyBatis是一个支持普通SQL查询,存储过程和高级映射的优 ...

  2. 笔记56 Mybatis快速入门(七)

    相关概念介绍(二) 6.一级缓存 <1>在一个session里查询相同id的数据 package mybatis.annotation; import java.io.IOExceptio ...

  3. 笔记50 Mybatis快速入门(一)

    一.Mybatis简介 MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis ...

  4. mybatis快速入门(四)

    mybatis动态标签<where><if><foreach>以及sql片段 1.创建一个包装类UserQueryVo.java package cn.my.myb ...

  5. 笔记55 Mybatis快速入门(六)

    相关概念介绍(一) 1.日志 有时候需要打印日志,知道mybatis执行了什么样的SQL语句,以便进行调试.这时,就需要开启日志,而mybatis自身是没有带日志的,使用的都是第三方日志,这里介绍如何 ...

  6. 笔记54 Mybatis快速入门(五)

    Mybatis中注解的使用 1.XML方式的CRUD 新增加接口CategoryMapper ,并在接口中声明的方法上,加上注解对比配置文件Category.xml,其实就是把SQL语句从XML挪到了 ...

  7. 笔记52 Mybatis快速入门(三)

    一.更多查询 1.模糊查询 修改Category.xml,提供listCategoryByName查询语句select * from category where name like concat(' ...

  8. 笔记51 Mybatis快速入门(二)

    Mybatis的CRUD 1.修改配置文件Category.xml,提供CRUD对应的sql语句. <?xml version="1.0" encoding="UT ...

  9. MyBatis学习总结(一)——MyBatis快速入门

    一.Mybatis介绍 MyBatis是一个支持普通SQL查询,存储过程和高级映射的优秀持久层框架.MyBatis消除了几乎所有的JDBC代码和参数的手工设置以及对结果集的检索封装.MyBatis可以 ...

随机推荐

  1. android中的BroadCastReceiver

    BroadCastReceiver组件本质是一种全局的监听器,用于监听系统全局的广播消息.由于BroadCastReceiver是一种全局的监听器,因此他可以非常方便的实现系统之间不同组件之间的通信. ...

  2. Codeforces542E Playing on Graph 思维+DFS+BFS

    解法参考https://www.cnblogs.com/BearChild/p/7683114.html这位大佬的,这位大佬讲得很好了. 这道题还是有一定的思维的. 直接贴代码: #include&l ...

  3. Java中使用try-catch-finally处理IO流中的异常

    我们使用try-catch-finally来接收IO流的异常 finally是最后执行的步骤,非常适合最后存放close来关闭IO流,而且编程中我们不可以随意抛出异常,必须对异常进行处理. 从try- ...

  4. IO流分类

    IO流在java中从输入和输出角度分类: 1.输入流 2.输出流 IO流在java中从数据的角度来分类: 1.字符流:文本,我们能读懂的都可以认为是字符流,如:文章,java文件等 字符输入流的超类: ...

  5. Windows 屏幕保护程序

    { 创建一个win32 窗口项目,不是控制台的 把exe改为src文件 复制到windows目录下 ok }

  6. obj.offsetHeight与obj.style.height $(obj).height()与$(obj).css('height')

    相同:都可以获取obj的高度区别:(1)obj.offsetHeight可以获取外部.内嵌和内联中定义的高,而obj.style.height只能获取内联中定义的高:(2)obj.offsetHeig ...

  7. J2EE学习篇之--JQuery技术详解

    前面我们讲解了的J2EE的技术都是服务端的技术,下面我们来看一下前端的一些开发技术,这一篇我们来看一下jQuery技术 简介: jQuery由美国人John Resig创建,至今已吸引了来自世界各地的 ...

  8. 模数循环节——cf547A

    campjls讲过模数循环节的问题,今天做cf才做到这类题 h1->a1的长度为len1,a1->a1的长度为cir1 h2->a2的长度为len2,a2->a2的长度为cir ...

  9. 浅谈学习selenium的一些知识点的总结

    学习自动化测试,先得学习一门语言.自动化对语言要求掌握的程度不深,但必须得会基本的入门语法. 我学习的是python2,简单,易懂,上手快. 每天敲就是了. 我的学习路径是: 先学习一段时间pytho ...

  10. 动态队列实现-----C语言

    /***************************************************** Author:Simon_Kly Version:0.1 Date: 20170520 D ...