热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

如何理解MyBatis动态SQL

本篇内容主要讲解“如何理解MyBatis动态SQL”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何理解M

本篇内容主要讲解“如何理解MyBatis动态SQL”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何理解MyBatis动态SQL”吧!

目录
  • 前言

  • 动态sql

    • 1.先看一下模块目录结构

    • 2.物理建模和逻辑建模

    • 3. 引入依赖

    • 4.全局配置文件

    • 5.sql共性抽取文件

    • 6.mapper接口

  • if

    • 静态sql:

    • 动态sql:

  • where

    • 用if标签的动态sql:

    • where和if的动态sql:

  • trim

    • trim的动态sql

    • trim标签:

  • set

    • trim的动态sql:

    • set的动态sql

    • set标签的作用:

  • choose、when、otherwise

    • 动态sql

  • foreach

    • 1.动态sql

    • 2.动态sql

    • 批量查询:foreach标签

  • 测试程序

    前言

    前面mysql都是通过静态sql进行查询的,但是如果业务复杂的时候,我们会遇到引号问题,或者多一个空格,这就使得sql代码编写错误了,所以为了解决这个问题,我们有了动态sql。

    Mybatis框架的动态SQL技术是一种根据特定条件动态拼装SQL语句的功能,它存在的意义是为了解决拼接SQL语句字符串时的痛点问题。具体是通过标签来实现的。

    动态sql

    1.先看一下模块目录结构

    在类路径的resources下的mapper包下创建sql.xml文件(共性抽取)

    如何理解MyBatis动态SQL

    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如下:

    
    
        4.0.0
    
        org.example
        day03-mybatis02-dynamic
        1.0-SNAPSHOT
        jar
    
        
            
                org.projectlombok
                lombok
                1.18.8
                provided
            
    
            
            
                org.mybatis
                mybatis
                3.5.7
            
    
            
            
                junit
                junit
                4.12
                test
            
    
            
            
                mysql
                mysql-connector-java
                5.1.3
                runtime
            
    
            
            
                log4j
                log4j
                1.2.17
            
        
    
    

    4.全局配置文件

    
    
    
        
        
            
        
        
        
            
        
        
        
            
                
                
                    
                    
                    
                    
                
                    
            
        
        
        
            
            
        
    

    注意: 这里有驼峰映射,别名映射,路径映射和路径映射。和以前的不同的是,我们这里做了sql语句的共性抽取,所以得加一个sql的路径映射

    5.sql共性抽取文件

    在类路径resources下的包mapper下创建一个sql.xml(因为我们sql是要写在映射文件中,自己本身也是映射文件,所以需要写在mapper下)。到要用的时候,在映射路径文件中需要用到这个sql语句的地方加入

    
    
    
    
            
        select emp_id,emp_name,emp_salary from t_emp
    
    
    
    

    共性抽取文件也可以不配置,这时候直接在映射文件中把要执行的语句重新编写就行了。

    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 selectEmployeeListByEmpId(Integer empId);
    
        /**
         * 查询大于传入的empId并且工资大于传入的empSalary的员工集合,如果传入的empId为null,则不考虑empId条件
         * 传入的empSalary为null则不考虑empSalary的条件
         */
        List selectEmployeeListByEmpIdAndEmpSalary(@Param("empId") Integer empId, @Param("empSalary") Double empSalary);
    
        /**
         * 根据empId更新员工信息,如果某个值为null,则不更新这个字段
         */
        void updateEmployee(Employee employee);
    
        /**
         * 根据emp_id查询员工信息,如果0 selectEmployeeList(Integer empId);
    
        /**
         * 添加员工信息 
         */
        void insertEmployee(Employee employee);
    
        /**
         * 批量添加员工集合    
         */
        void insertEmployeeList(@Param("employeeList") List employeeList);
    
        /**
         * 根据员工的id集合查询员工集
         */
        List selectEmployeeListByEmpIdList(List idList);
    }

    if

    目标:根据员工的empId查询大于该empId的所有员工,如果empId为null,则查询全体员工。

    Dao接口的方法为:
    List selectEmployeeListByEmpId(Integer empId);

    静态sql:

    
    
         where emp_id>#{empId}
    
    

    动态sql:

    
         
         
             where emp_id>#{empId}
         
     

    表示引用抽取出的sql片段,也可以直接写sql语句。如果是静态sql,当id为null时,查询出来的是空,动态sql则可以查出全部。if标签里面有test属性名,作为判断语句。

    where

    目标:

    • 查询大于传入的empId并且工资大于传入的empSalary的员工集合

    • 如果传入的empId为null,则不考虑empId条件

    • 传入的empSalary为null则不考虑empSalary的条件

    Dao接口方法:

    List selectEmployeeListByEmpIdAndEmpSalary(@Param("empId") Integer empId, @Param("empSalary") Double empSalary);

    用if标签的动态sql:

    
         where
       
                emp_id>#{empId}
            
            
               and emp_salary>#{empSalary}
            

    这里可以看到,如果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:

    
        
       
        
            
                emp_id>#{empId}
            
            
               and emp_salary>#{empSalary}
            
        
    

    where标签的作用:

    • 在第一个条件之前自动添加WHERE关键字

    • 自动去掉第一个条件前的连接符(AND、OR等等)

    trim

    trim是修建的意思,其实就是去头去尾,这里还是根据上面那个方法

    trim的动态sql

    
        
        
            
                emp_id>#{empId}
            
    
            
                AND emp_salary>#{empSalary}
            
        
    

    trim标签:

    • prefix:指定要动态添加的前缀

    • suffix属性:指定要动态添加的后缀

    • prefixOverrides:指定要动态去掉的前缀,使用“|”分隔有可能的多个值

    • suffixOverrides属性:指定要动态去掉的后缀,使用“|”分隔有可能的多个值

    set

    目标:根据empId更新员工信息,如果某个值为null,则不更新这个字段

    Dao接口方法:
    void updateEmployee(Employee employee);
    我们先用上面的trim标签来解决一下这个问题,

    trim的动态sql:

    
        
            
                emp_name=#{empName}
            
            
                , emp_salary=#{empSalary}
            
        
        where emp_id=#{empId}
    

    set的动态sql

    
        update t_emp
         
             
                 emp_name=#{empName}
             
             
                , emp_salary=#{empSalary}
             
         

    可以看出

    set标签的作用:

    • 自动在要修改的第一个字段之前添加SET关键字

    • 去掉要修改的第一个字段前的连接符(,)

    choose、when、otherwise

    目标:

    • 根据emp_id查询员工信息,如果0

    • 如果emp_id是大于6,那么就查询所有小于该emp_id的员工

    • 如果是其它情况,则查询所有员工信息

    Dao接口方法:
    List selectEmployeeList(Integer empId);

    动态sql

    
      
         where
        
        
            0 and empId<6">
                emp_id>#{empId}
            
            6">
                emp_id<#{empId}
            
            
                1==1
            
        
    
    

    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 into t_emp(emp_name,emp_salary)values
        
        
            (#{emp.empName},#{emp.empSalary})
        
    

    目标2:根据多个id查询多个员工信息

    Dao接口

    List selectEmployeeListByEmpIdList(List idList);

    2.动态sql

    
        
         
             #{id}
         
    

    批量查询:foreach标签

    1. collection属性: 表示要遍历的对象,如果要遍历的参数使用@Param注解取名了就使用该名字,如果没有取名List,或者collection。

    2. item属性: 表示遍历出来的元素,我们到时候要拼接SQL语句就得使用这个元素: 如果遍历出来的元素是POJO对象, 那么我们就通过 #{遍历出来的元素.POJO的属性} 获取数据;如果遍历出来的元素是简单类型的数据,那么我们就使用 #{遍历出来的元素} 获取这个简单类型数据

    3. separator属性: 遍历出来的元素之间的分隔符

    4. open属性: 在遍历出来的第一个元素之前添加前缀

    5. 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 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 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使用
      本文介绍了MyBatis多表查询与动态SQL的使用方法,包括一对一查询和一对多查询。同时还介绍了动态SQL的使用,包括if标签、trim标签、where标签、set标签和foreach标签的用法。文章还提供了相关的配置信息和示例代码。 ... [详细]
    • 学习mybatis的基础知识:mybatis入门教程(二)
      2019独角兽企业重金招聘Python工程师标准2.3MyBatisprintsql在log4j.properties配置文件中添加如下配置,让mybatis打 ... [详细]
    • 是的,可以在RDSMySqlEngine中进行Master-Master复制。但它需要一些操作与实例。先决条件:1)为启用二进制日志记录创建两个实例的只 ... [详细]
    • 问题说明最近看到Spring事务,在学习过程中遇到一个很苦恼问题搭建好Spring的启动环境后出现了一点小问题在启动时候却出现[java.lang.NullPointerExcep ... [详细]
    • 先看看效果是不是自己想要的吧item及item内部控件点击事件不懂的可以先点击查看 ... [详细]
    • 如何实现Percona Mysql Galera多读写集群的部署
      本篇文章给大家主要讲的是关于如何实现PerconaMysqlGalera多读写集群的部署的内容,感兴趣的话就一起来看看这篇文章吧,相信看完如何实现PerconaMysq ... [详细]
    • 代码如下:(把数据以表格形式输出本人小白哪位大侠能把完整的代码写出来呀最好是可以添加、删除和显示表格的)<%@pagecontentTypetexthtmlpageEn ... [详细]
    • 关于初学PHP时的知识积累总结【PHP】
      后端开发|php教程PHP,知识积累后端开发-php教程PHP基础A、初识PHPPHP是与HTML混合使用的嵌入式语言。1、PHP标记默认标记短标记,需在php.ini中将shor ... [详细]
    • 这篇文章主要介绍“大文本数据怎么导入导出到数据库”,在日常操作中,相信很多人在大文本数据怎么导入导出到数据库问题上存在疑惑,小编查阅了各 ... [详细]
    • 本文介绍了使用kotlin实现动画效果的方法,包括上下移动、放大缩小、旋转等功能。通过代码示例演示了如何使用ObjectAnimator和AnimatorSet来实现动画效果,并提供了实现抖动效果的代码。同时还介绍了如何使用translationY和translationX来实现上下和左右移动的效果。最后还提供了一个anim_small.xml文件的代码示例,可以用来实现放大缩小的效果。 ... [详细]
    • Spring特性实现接口多类的动态调用详解
      本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
    • eclipse学习(第三章:ssh中的Hibernate)——11.Hibernate的缓存(2级缓存,get和load)
      本文介绍了eclipse学习中的第三章内容,主要讲解了ssh中的Hibernate的缓存,包括2级缓存和get方法、load方法的区别。文章还涉及了项目实践和相关知识点的讲解。 ... [详细]
    • XML介绍与使用的概述及标签规则
      本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
    • springboot dubbo框架中log4j与slf4jlog4j12日志冲突问题的解决方法
      在基于springboot开发项目中,使用dubbo的RPC框架进行业务拆分,出 ... [详细]
    • 介绍怎样在IntellijIdea中通过创建mavenproject配置MapReduce的编程环境。一、软件环境我使用的软件版本号例如以下:IntellijIdea2017.1M ... [详细]
    author-avatar
    隆威SU_486
    这个家伙很懒,什么也没留下!
    PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
    Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有