Java MyBatis3(8)动态SQL标签

前言:

mybatis框架中最具特色的便是sql语句中的自定义,而动态sql的使用又使整个框架更加灵活。

创建User表

/*Table structure for table `user` */

DROP TABLE IF EXISTS `user`;

CREATE TABLE `user` (
  `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
  `name` varchar(20) NOT NULL,
  `username` varchar(20) NOT NULL,
  `age` int(3) NOT NULL,
  `phone` varchar(11) NOT NULL,
  `email` varchar(50) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8;

/*Data for the table `user` */

insert  into `user`(`id`,`name`,`username`,`age`,`phone`,`email`) values (1,'张三','zs',18,'15010998046','4567899@qq.com'),(2,'李四','ls',19,'15019087600','567657642@qq.com'),(3,'王五','ww',20,'15010898065','2323248@qq.com');

/*!40101 SET SQL_MODE=@OLD_SQL_MODE */;
/*!40014 SET FOREIGN_KEY_CHECKS=@OLD_FOREIGN_KEY_CHECKS */;
/*!40014 SET UNIQUE_CHECKS=@OLD_UNIQUE_CHECKS */;
/*!40111 SET SQL_NOTES=@OLD_SQL_NOTES */;
View Code

标签  作用
if  单条件分支,相当于判断语句
choose、when、otherwise 多条件分支,相当于Java中的switch语句
set、where、trim 辅助条件判断,用于拼接语句
foreach  集合进行遍历(尤其是在构建 IN 条件语句的时候)
bind  创建一个变量,并将其绑定到当前的上下文

 

if标签

User实体

package cn.cnki.ref.pojo;

public class User {

    private int id;
    private String name;
    private String username;
    private int age;
    private String phone;
    private String email;
    //无参构造函数必须有,ORM框架调用的就是无参构造函数
    public User() {

    }
    public User(int id, String name, String username, int age, String phone, String email) {
        super();
        this.id = id;
        this.name = name;
        this.username = username;
        this.age = age;
        this.phone = phone;
        this.email = email;
    }
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username = username;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public String getPhone() {
        return phone;
    }
    public void setPhone(String phone) {
        this.phone = phone;
    }
    public String getEmail() {
        return email;
    }
    public void setEmail(String email) {
        this.email = email;
    }

    @Override
    public String toString() {
        return "User [id=" + id + ", name=" + name + ", username=" + username + ", age=" + age + ", phone=" + phone
                + ", email=" + email + "]";
    }
}
View Code

UserMapper

package cn.cnki.ref.mapper;

import cn.cnki.ref.pojo.User;

import java.util.List;

public interface UserMapper {
    public List<User> getUser(User user);
}
View Code

UserMapper.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关系映射  -->
<mapper namespace="cn.cnki.ref.mapper.UserMapper">

    <select id="getUser" resultType="cn.cnki.ref.pojo.User">
        select * from user where 1=1
        <if test="id != null">and id=#{id}</if>
        <if test="age != null">and age=#{age}</if>
        <if test="name != null">and name=#{name}</if>
    </select>

</mapper>
View Code

UserController

package cn.cnki.ref.controller;

import cn.cnki.ref.mapper.UserMapper;
import cn.cnki.ref.pojo.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.List;

@RestController
public class UserController {
    @Autowired
    private UserMapper UserMapper;

    @GetMapping("/getUser")
    public List<User> getUser() {
        User user = new User(1, "张三", null, 18, null, null);
        List<User> users = UserMapper.getUser(user);
        System.out.println(users);
        return users;
    }

}
View Code

测试

http://localhost:8080/getUser

choose (when, otherwise)标签

  choose标签是按顺序判断其内部when标签中的test条件出否成立,如果有一个成立,则 choose 结束。当 choose 中所有 when 的条件都不满则时,则执行 otherwise 中的sql。类似于Java 的 switch 语句,choose 为 switch,when 为 case,otherwise 则为 default。

  例如下面例子,同样把所有可以限制的条件都写上,方面使用。choose会从上到下选择一个when标签的test为true的sql执行。安全考虑,我们使用where将choose包起来,放置关键字多于错误。

<!--  choose(判断参数) - 按顺序将实体类 User 第一个不为空的属性作为:where条件 -->  
<select id="getUserList_choose" resultMap="resultMap_user" parameterType="com.yiibai.pojo.User">  
    SELECT *  
      FROM User u   
    <where>  
        <choose>  
            <when test="username !=null ">  
                u.username LIKE CONCAT(CONCAT('%', #{username, jdbcType=VARCHAR}),'%')  
            </when >  
            <when test="sex != null and sex != '' ">  
                AND u.sex = #{sex, jdbcType=INTEGER}  
            </when >  
            <when test="birthday != null ">  
                AND u.birthday = #{birthday, jdbcType=DATE}  
            </when >  
            <otherwise>  
            </otherwise>  
        </choose>  
    </where>    
</select>
View Code

where标签

<select id="getUser" resultType="cn.cnki.ref.pojo.User">
        select * from user
        <where>
            <if test="id != null">and id=#{id} </if>
            <if test="age != null">and age=#{age} </if>
            <if test="name != null">and name=#{name}</if>
        </where>
    </select>
View Code

输出sql语句

MyBatis 中为什么不建议使用 where 1=1?

在 MyBatis 中,建议尽量避免使用无意义的 SQL 拼接 where 1=1,我们可以使用 标签来替代 where 1=1,这样的写既简洁又优雅,何乐而不为呢?

<?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.example.demo.mapper.UserMapper">
    <select id="list" resultType="com.example.demo.model.User">
        select * from user
        <where>
            <if test="name!=null">
               and name=#{name}
            </if>
            <if test="password!=null">
                and password=#{password}
            </if>
        </where>
    </select>
</mapper>

trim标签

 <select id="getUser" resultType="cn.cnki.ref.pojo.User">
        select * from user
        <trim prefix="where" suffixOverrides="and">
            <if test="id != null">id=#{id} and</if>
            <if test="age != null">age=#{age} and</if>
            <if test="name != null">name=#{name} and</if>
        </trim>
    </select>
View Code

trim标签下的四个属性:

  prefix:在标签开始添加上该字符串

  suffixOverrides:在标签末尾去除上该字符串

  suffix:在标签末尾添加上该字符串

  prefixOverrides:在标签开始去除上该字符串

set标签

   用在update语句中,如果字段参数不为null,则修改此参数

UserMapper

public interface UserMapper {
    public int updateUserById(User user);
}
View Code

UserMapper.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关系映射  -->
<mapper namespace="cn.cnki.ref.mapper.UserMapper">

    <update id="updateUserById">
        update user
        <set>
            <if test="name != null">name=#{name},</if>
            <if test="age != null">age=#{age},</if>
            <if test="username != null">username=#{username},</if>
            <if test="email != null">email=#{email},</if>
            <if test="phone != null">phone=#{phone}</if>
        </set>
        <where>
            id=#{id}
        </where>
    </update>

</mapper>
View Code

UserController

package cn.cnki.ref.controller;

import cn.cnki.ref.mapper.UserMapper;
import cn.cnki.ref.pojo.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.List;

@RestController
public class UserController {
    @Autowired
    private UserMapper UserMapper;
    
    @GetMapping("/updateUserById")
    public int updateUserById() {
        User user = new User(1, null, null, 21, "12545564454", "14548445@qq.com");
        return  UserMapper.updateUserById(user);
    }
}
View Code

测试结果

switch\when标签

1描述需求:

  满足id!=null查询id,

  否则,看满足age否,

  接着,看name是否满足,

  最后,按age>10查询

2.改mapper映射

<select id="getUser" resultType="cn.cnki.ref.pojo.User">
        select * from user where
        <choose>
            <when test="id != null">id=#{id} </when>
            <when test="age != null">age=#{age}</when>
            <when test="name != null">name=#{name} </when>
            <otherwise>age>10</otherwise>
        </choose>
    </select>
View Code

3.测试结果

foreach标签

1.需求描述:

  查出多个id的user

2.添加mapper接口

 public List<User> listById(List<Integer> ids);

3添加mapper映射

<select id="listById" resultType="cn.cnki.ref.pojo.User" parameterType="list">
        select * from user where id in
        <foreach collection="list" item="id" separator="," open="(" close=")">
            #{id}
        </foreach>
    </select>
View Code

4.测试

 @GetMapping("/listById")
    public List<User> listById() {
        List<User> users = UserMapper.listById(Arrays.asList(1,2));
        System.out.println(users);
        return users;
    }
View Code

5.foreach标签下的所有属性:

  collection:获取的集合名,如果是list集合,springboot会把它的key值默认封装为list

  item:遍历的单个属性值

  separator:拼接隔离的字符串

  open:在循环的开始拼接的字符串

  close:在循环的结束拼接的字符串

  index:索引,在map中作为key

bind

bind标签可以使用OGNL表达式创建一个变量并将其绑定到上下文中。

使用concat函数连接字符串,在MySQL中,这个函数支持多个参数,但是在Oracle中只支持两个参数。 由于不同数据库之间的语法差异,如果更换了数据库,有些SQL语句可能就需要重写。 针对这种情况,可以使用bind标签来避免由于更换数据库带来的一些麻烦。

posted @ 2018-07-21 13:11  ~沐风  阅读(514)  评论(0编辑  收藏  举报