MyBatis学习总结

第一个 MyBatis 程序

我们来实际开发一个 MyBatis 程序,感受一下。

第一步:准备数据库

首先我们创建一个数据库【mybatis】,编码方式设置为 UTF-8,然后再创建一个名为【student】的表,插入几行数据:

DROP DATABASE IF EXISTS mybatis;
CREATE DATABASE mybatis DEFAULT CHARACTER SET utf8;

use mybatis;
CREATE TABLE student(
  id int(11) NOT NULL AUTO_INCREMENT,
  studentID int(11) NOT NULL UNIQUE,
  name varchar(255) NOT NULL,
  PRIMARY KEY (id)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

INSERT INTO student VALUES(1,1,'我没有三颗心脏');
INSERT INTO student VALUES(2,2,'我没有三颗心脏');
INSERT INTO student VALUES(3,3,'我没有三颗心脏');

 

第二步:创建工程

在 IDEA 中新建一个 Java 工程,并命名为【HelloMybatis】并引入相应的坐标

    jar

    
        
            org.mybatis
            mybatis
            3.4.5
        

        
            mysql
            mysql-connector-java
            5.1.6
        

        
            log4j
            log4j
            1.2.12
        

        
            junit
            junit
            4.10
        
    


 

第三步:创建实体类

在 Package【pojo】下新建实体类【Student】,用于映射表 student:

package pojo;

public class Student {

    int id;
    int studentID;
    String name;

    /* getter and setter */
}

 

第四步:配置文件 mybatis-config.xml

在【src】目录下创建 MyBaits 的主配置文件 mybatis-config.xml ,其主要作用是提供连接数据库用的驱动,数据名称,编码方式,账号密码等,我们在后面说明:





    
    
        <package name="pojo"/>
    
    
    default="development">
        
            
            
                
                
                
                
            
        
    
    
    
        
    

 

第五步:配置文件 Student.xml

在 Package【pojo】下新建一个【Student.xml】文件:


DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">


    

 

  • 由于上面配置了  别名,所以在这里的 resultType 可以直接写 Student,而不用写类的全限定名 pojo.Student
  • namespace 属性其实就是对 SQL 进行分类管理,实现不同业务的 SQL 隔离
  • SQL 语句的增删改查对应的标签有:

第六步:编写测试类

在 Package【test】小创建测试类【TestMyBatis】:

package test;

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 pojo.Student;

import java.io.IOException;
import java.io.InputStream;
import java.util.List;

public class TestMyBatis {

    public static void main(String[] args) throws IOException {
        // 根据 mybatis-config.xml 配置的信息得到 sqlSessionFactory
        String resource = "mybatis-config.xml";
        InputStream inputStream = Resources.getResourceAsStream(resource);
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        // 然后根据 sqlSessionFactory 得到 session
        SqlSession session = sqlSessionFactory.openSession();
        // 最后通过 session 的 selectList() 方法调用 sql 语句 listStudent
        List listStudent = session.selectList("listStudent");
        for (Student student : listStudent) {
            System.out.println("ID:" + student.getId() + ",NAME:" + student.getName());
        }

    }
}

 

运行测试类:

MyBatis学习总结_第1张图片

基本原理

  • 应用程序找 MyBatis 要数据
  • MyBatis 从数据库中找来数据
    1.通过 mybatis-config.xml 定位哪个数据库
    2.通过 Student.xml 执行对应的 sql 语句
    3.基于 Student.xml 把返回的数据库封装在 Student 对象中
    4.把多个 Student 对象装载一个 Student 集合中
  • 返回一个 Student 集合

MyBatis学习总结_第2张图片

参考资料:How2j.cn-MyBatis 相关教程


CRUD 操作

我们来看看常规的一套增删改查应该怎么实现:

第一步:配置 Student.xml

首先,我们在 SQL 映射文件中新增语句,用来支撑 CRUD 的系列操作


DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">


    

    
        insert into student (id, studentID, name) values (#{id},#{studentID},#{name})
    

    
        delete from student where id = #{id}
    

    

    
        update student set name=#{name} where id=#{id}
    

 

  • parameterType:要求输入参数的类型
  • resultType:输出的类型

第二步:实现增删改查

package test;

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 pojo.Student;

import java.io.IOException;
import java.io.InputStream;
import java.util.List;

public class TestMyBatis {

    public static void main(String[] args) throws IOException {
        // 根据 mybatis-config.xml 配置的信息得到 sqlSessionFactory
        String resource = "mybatis-config.xml";
        InputStream inputStream = Resources.getResourceAsStream(resource);
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        // 然后根据 sqlSessionFactory 得到 session
        SqlSession session = sqlSessionFactory.openSession();

        // 增加学生
        Student student1 = new Student();
        student1.setId(4);
        student1.setStudentID(4);
        student1.setName("新增加的学生");
        session.insert("addStudent", student1);

        // 删除学生
        Student student2 = new Student();
        student2.setId(1);
        session.delete("deleteStudent", student2);

        // 获取学生
        Student student3 = session.selectOne("getStudent", 2);

        // 修改学生
        student3.setName("修改的学生");
        session.update("updateStudent", student3);

        // 最后通过 session 的 selectList() 方法调用 sql 语句 listStudent
        List listStudent = session.selectList("listStudent");
        for (Student student : listStudent) {
            System.out.println("ID:" + student.getId() + ",NAME:" + student.getName());
        }

        // 提交修改
        session.commit();
        // 关闭 session
        session.close();
    }
}

 

上述的程序中:

  • 通过 session.insert("addStudent", student1); 增加了一个 ID 和 studentID 都为 4,名字为“新增加的学生” 的学生
  • 通过 session.delete("deleteStudent", student2); 删除了 ID = 1 的学生
  • 通过 Student student3 = session.selectOne("getStudent", 2); 获取了 ID = 2的学生
  • 通过 session.update("updateStudent", student3); 将 ID = 2 的学生的名字修改为 “修改的学生”
  • 通过 session.commit() 来提交事务,也可以简单理解为更新到数据库

运行获得正确结果:

MyBatis学习总结_第3张图片

模糊查询

如果要对数据库中的 student 表进行模糊查询,需要通过匹配名字中的某个字来查询该用户。

我们首先在 Student.xml 配置文件中配置 SQL 映射:

 

  • 注意:  类型的查询语句 SQL 配置,将之前设计好的 SQL 语句配置进去,然后指定输出参数属性为 resultType,类型为 StudentAndCard 这个 Java 包装类:

     

    然后在测试类中编写测试方法:

    @Test
    public void test() throws IOException {
    
        // 根据 mybatis-config.xml 配置的信息得到 sqlSessionFactory
        String resource = "mybatis-config.xml";
        InputStream inputStream = Resources.getResourceAsStream(resource);
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        // 然后根据 sqlSessionFactory 得到 session
        SqlSession session = sqlSessionFactory.openSession();
    
        // 找到身份证身份证号码为 1111 的学生
        StudentAndCard student = session.selectOne("findStudentByCard",1111);
        // 获得其姓名并输出
        System.out.println(student.getName());
    }

     

    获得正确结果:

    MyBatis学习总结_第7张图片

    2. 使用 resultMap 实现

    使用 resultMap 可以将数据字段映射到名称不一样的响应实体类属性上,重要的是,可以映射实体类中包裹的其他实体类。

    首先我们来创建一个封装了 Card 号码和 Student 实体类的 StudentWithCard 类:

    package pojo;
    
    public class StudentWithCard {
        
        Student student;
        int number;
        int id;
    
        /* getter and setter */
    }

     

    SQL 语句依然没有变化,但是使用的输出映射属性改为了 resultMap ,其中的映射类型是 id 为 StudentInfoMap 的 resultMap 配置:

    
    
    
        
        
        
        
        
            
            
            
        
    

     

    稍微修改一下测试类,测试使用 resultMap 实现的一对一查询映射:

    @Test
    public void test() throws IOException {
    
        // 根据 mybatis-config.xml 配置的信息得到 sqlSessionFactory
        String resource = "mybatis-config.xml";
        InputStream inputStream = Resources.getResourceAsStream(resource);
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        // 然后根据 sqlSessionFactory 得到 session
        SqlSession session = sqlSessionFactory.openSession();
    
        // 找到身份证身份证号码为 1111 的学生
        StudentWithCard student = session.selectOne("findStudentByCard", 1111);
        // 获得其姓名并输出
        System.out.println(student.getStudent().getName());
    }

     

    测试仍然能得到正确的结果:

    MyBatis学习总结_第8张图片

    一对多查询

    MyBatis学习总结_第9张图片

    还是先来建立数据模型,删掉之前的:

    use mybatis;
    CREATE TABLE student (
      student_id int(11) NOT NULL AUTO_INCREMENT,
      name varchar(255) DEFAULT NULL,
      PRIMARY KEY (student_id)
    )AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
    
    CREATE TABLE class (
      class_id int(11) NOT NULL AUTO_INCREMENT,
      name varchar(255) NOT NULL,
      student_id int(11)  NOT NULL,
      PRIMARY KEY (class_id)
    )AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
    
    INSERT INTO student VALUES (1,'student1');
    INSERT INTO student VALUES (2,'student2');
    
    INSERT INTO class VALUES (1,'Java课',1);
    INSERT INTO class VALUES (2,'Java课',2);

     

    • 其中 class 的 name 字段表示课程的名称。

    然后我们来编写我们的 SQL 语句:

    SELECT 
      student.*
    FROM
      student, class
    WHERE student.student_id = class.student_id AND class.class_id = #{value}

     

    我们执行的结果如下:

    MyBatis学习总结_第10张图片

    我们再来创建对应的实体类:

    public class Student {
    
        private int id;
        private String name;
    
        /* getter and setter */
    }
    
    public class Class {
    
        private int id;
        private String name;
        private List students;
    
        /* getter and setter */
    }

     

    在 Package【pojo】下新建一个【class.xml】文件完成配置:

    
    DOCTYPE mapper
            PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
            "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
    
    
        
            
            
        
        
    

     

    编写测试类:

    @Test
    public void test() throws IOException {
    
        // 根据 mybatis-config.xml 配置的信息得到 sqlSessionFactory
        String resource = "mybatis-config.xml";
        InputStream inputStream = Resources.getResourceAsStream(resource);
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        // 然后根据 sqlSessionFactory 得到 session
        SqlSession session = sqlSessionFactory.openSession();
    
        // 查询上Java课的全部学生
        List students = session.selectList("listStudentByClassName", "Java课");
        for (Student student : students) {
            System.out.println("ID:" + student.getId() + ",NAME:" + student.getName());
        }
    }

     

    运行测试结果,成功:

    MyBatis学习总结_第11张图片

    多对多查询

    MyBatis学习总结_第12张图片

    建立数据模型:

    use mybatis;
    CREATE TABLE students (
      student_id int(11) NOT NULL AUTO_INCREMENT,
      student_name varchar(255) DEFAULT NULL,
      PRIMARY KEY (student_id)
    )AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
    
    CREATE TABLE courses (
      course_id int(11) NOT NULL AUTO_INCREMENT,
      course_name varchar(255) NOT NULL,
      PRIMARY KEY (course_id)
    )AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
    
    CREATE TABLE student_select_course(
      s_id int(11) NOT NULL,
      c_id int(11) NOT NULL,
      PRIMARY KEY(s_id,c_id)
    ) DEFAULT CHARSET=utf8;
    
    INSERT INTO students VALUES (1,'student1');
    INSERT INTO students VALUES (2,'student2');
    
    INSERT INTO courses VALUES (1,'Java课');
    INSERT INTO courses VALUES (2,'Java Web课');
    
    INSERT INTO student_select_course VALUES(1,1);
    INSERT INTO student_select_course VALUES(1,2);
    INSERT INTO student_select_course VALUES(2,1);
    INSERT INTO student_select_course VALUES(2,2);

     

    根据要求我们来设计一下 SQL 语言:

    SELECT
        s.student_id,s.student_name
    FROM
        students s,student_select_course ssc,courses c
    WHERE s.student_id = ssc.s_id 
    AND ssc.c_id = c.course_id 
    AND c.course_name = #{value}

     

    执行 SQL 结果如下:

    MyBatis学习总结_第13张图片

    实体类雷同,就不再赘述,我们直接来配置映射文件【Student.xml】:

    
        
        
    
    
    

     

    测试类也雷同,只需要修改一下调用的 id (改为findStudentsByCourseName)就好了,直接上测试结果:

    MyBatis学习总结_第14张图片

    相反也是一样的,重要的是 SQL 语句和映射。

    总结:

    • 自己写的 SQL 语句看着虽然没有很恶心(至少思路清晰),但感觉很烂!
    • 结合 SQL 语言和映射文件,能够很方便的操作数据库
    • 数据库还是建立外键得好....(啪啪打脸,根据《阿里Java开发手册》里提到,最好不要建外键,而让程序的Service层去做判断)

    延迟加载

    什么是延迟加载?从字面上理解,就是对某一类信息的加载之前需要延迟一会儿。在 MyBatis 中,通常会进行多表联合查询,但是有的时候不会立即用到所有的联合查询结果,这时候就可以采用延迟加载的功能。

    • 功能: 延迟加载可以做到,先从单表查询,需要时再从关联表关联查询,这样就大大提高了数据库的性能,因为查询单表要比关联查询多张表速度快。
    • 实例: 如果查询订单并且关联查询用户信息。如果先查询订单信息即可满足要求,当我们需要查询用户信息时再查询用户信息。把对用户信息的按需去查询就是延迟加载。
    关联查询:
    SELECT 
        orders.*, user.username 
    FROM orders, user
    WHERE orders.user_id = user.id
    延迟加载相当于:
    SELECT 
        orders.*,
        (SELECT username FROM USER WHERE orders.user_id = user.id)
        username 
    FROM orders

     

     

    所以这就比较直观了,也就是说,我把关联查询分两次来做,而不是一次性查出所有的。第一步只查询单表orders,必然会查出orders中的一个user_id字段,然后我再根据这个user_id查user表,也是单表查询。

    参考文章:[【MyBatis学习11】MyBatis中的延迟加载](https://blog.csdn.net/eson_15/article/details/51668523)

    Mapper 映射配置编写

    首先在 Mapper 映射文件中定义只查询所有订单信息的 SQL :

    <select id="findOrdersUserLazyLoading" resultMap="OrdersUserLazyLoadingResultMap">
        SELECT * FROM orders
    select>

     

    上面的 SQL 语句查询所有的订单信息,而每个订单信息中会关联查询用户,但由于希望延迟加载用户信息,所以会在 id 为 "OrdersUserLazyLoadingResultMap" 的 resultMap 对应的结果集配置中进行配置:

    MyBatis学习总结_第15张图片

    最后配置延迟加载要执行的获取用户信息的 SQL:

    <select id="findUserById" parameterType="int" resultType="user">
        select * from user where id = #{id}
    select>

     

    上面的配置会被用来延迟加载的 resultMap 中的 association 调用,输入参数就是 association 中 column 中定义的字段信息。

    在编写测试方法之前,首先需要开启延迟加载功能(这在 MyBatis 中默认是禁用掉的)。这需要在 MyBatis 的全局配置文件 mybatis-config.xml 中配置 setting 属性,将延迟加载(lazyLoadingEnable)的开关设置成 “ture” ,并且由于是按需加载,所以还需要将积极加载改为消极加载:

    <settings>
        <!-- 打开延迟加载的开关 -->
        <setting name="lazyLoadingEnabled" value="true"/>
        <!-- 将积极加载改为消极加载,即延迟加载 -->
        <setting name="aggressiveLazyLoading" value="false"/>
    settings>

     

    • 注意: 在 configuration 中配置是有一定顺序的,具体可以按住【Ctrl】不放点击 configuration 属性,能看到如下信息(即定义的顺序):
    <!ELEMENT configuration (properties?, settings?, typeAliases?, typeHandlers?, objectFactory?, objectWrapperFactory?, reflectorFactory?, plugins?, environments?, databaseIdProvider?, mappers?)>

    Mapper 动态代理

    什么是 Mapper 动态代理?一般创建 Web 工程时,从数据库取数据的逻辑会放置在 DAO 层(Date Access Object,数据访问对象)。使用 MyBatis 开发 Web 工程时,通过 Mapper 动态代理机制,可以只编写数据交互的接口及方法定义,和对应的 Mapper 映射文件,具体的交互方法实现由 MyBatis 来完成。这样大大节省了开发 DAO 层的时间。

    实现 Mapper 代理的方法并不难,只需要遵循一定的开发规范即可。

    Mapper 代理实例编写

    我们编写一个使用 Mapper 代理查询学生信息的示例,首先还是在【pojo】下新建一个名为 StudentMapper.xml 的 Mapper 配置文件,其中包含了对 Student 的增删改查的 SQL 配置:

    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.StudentMapper">
        
        <select id="findStudentById" parameterType="_int" resultType="pojo.Student">
            SELECT * FROM student WHERE student_id = #{id}
        select>
        
        <insert id="insertStudent" parameterType="pojo.Student">
            INSERT INTO student(student_id, name) VALUES(#{id}, #{name})
        insert>
        
        <delete id="deleteStudent" parameterType="_int">
            DELETE FROM student WHERE student_id = #{id}
        delete>
        
        <update id="updateStudent" parameterType="pojo.Student">
            UPDATE student SET name = #{name} WHERE student_id = #{id}
        update>
    mapper>

     

    如果需要使用 StudentMapper.xml 的 Mapper 代理,首先需要定义一个接口,名为 StudentMapper。然后在里面新建四个方法定义,分别对应 StudentMapper.xml 中的 Student 的增删改查的 SQL 配置,然后将 StudentMapper 中的 namespace 改为 StudentMapper 接口定义的地方(也就是 mapper 包下的 StudentMapper),这样就可以在业务类中使用 Mapper 代理了,接口代码如下:

    package mapper;
    
    import pojo.Student;
    
    public interface StudentMapper {
    
        // 根据 id 查询学生信息
        public Student findStudentById(int id) throws Exception;
    
        // 添加学生信息
        public void insertStudent(Student student) throws Exception;
    
        // 删除学生信息
        public void deleteStudent(int id) throws Exception;
    
        // 修改学生信息
        public void updateStudent(Student student) throws Exception;
    }

     

    • 注意: 别忘了在 mybatis-config.xml 中配置一下 Mapper 映射文件

    测试动态代理

    在测试方法中,使用 SqlSession 类的 getMapper 方法,并将要加载的 Mapper 代理的接口类传递进去,就可以获得相关的 Mapper 代理对象,使用 Mapper 代理对象去对学生信息进行增删改查:

    @Test
    public void test() throws Exception {
    
        // 根据 mybatis-config.xml 配置的信息得到 sqlSessionFactory
        String resource = "mybatis-config.xml";
        InputStream inputStream = Resources.getResourceAsStream(resource);
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        // 然后根据 sqlSessionFactory 得到 session
        SqlSession session = sqlSessionFactory.openSession();
        // 获取 Mapper 代理
        StudentMapper studentMapper = session.getMapper(StudentMapper.class);
        // 执行 Mapper 代理独享的查询方法
        Student student = studentMapper.findStudentById(1);
        System.out.println("学生的姓名为:" + student.getName());
        session.close();
    }

     

    运行测试方法,看到正确的结果:

    MyBatis学习总结_第16张图片

    使用 Mapper 代理可以让开发更加简洁,使查询结构更加清晰,工程结构更加规范。


    使用注解开发 MyBatis

    在上面的例子中,我们已经有了方便的 Mapper 代理对象,我们可以进一步省掉 XML 的配置信息,进而使用方便的注解来开发 MyBatis ,让我们实际来操练一下:

    第一步:为 Mapper 增加注解

    我们把 StudentMapper.xml 下配置的 SQL 语句通过注解的方式原封不动的配置在 StudentMapper 接口中:

    public interface StudentMapper {
    
        // 根据 id 查询学生信息
        @Select("SELECT * FROM student WHERE student_id = #{id}")
        public Student findStudentById(int id) throws Exception;
    
        // 添加学生信息
        @Insert("INSERT INTO student(student_id, name) VALUES(#{id}, #{name})")
        public void insertStudent(Student student) throws Exception;
    
        // 删除学生信息
        @Delete("DELETE FROM student WHERE student_id = #{id}")
        public void deleteStudent(int id) throws Exception;
    
        // 修改学生信息
        @Update("UPDATE student SET name = #{name} WHERE student_id = #{id}")
        public void updateStudent(Student student) throws Exception;
    }

     

    第二步:修改 mybatis-config.xml

    将之前配置的映射注释掉,新建一条:

    
    
        
        class="mapper.StudentMapper"/>
    

     

    • 注意: 这次映射的并不是文件(使用 resource 属性),而是类(使用 class 属性)

    第三步:运行测试代码

    上面的测试代码不用修改,直接运行,也能得到正确结果:

    MyBatis学习总结_第17张图片

    更多的注解:戳这里


    MyBatis 缓存结构

    MyBatis学习总结_第18张图片

    在 Web 系统中,最重要的操作就是查询数据库中的数据。但是有些时候查询数据的频率非常高,这是很耗费数据库资源的,往往会导致数据库查询效率极低,影响客户的操作体验。于是我们可以将一些变动不大且访问频率高的数据,放置在一个缓存容器中,用户下一次查询时就从缓存容器中获取结果。

    • MyBatis 拥有自己的缓存结构,可以用来缓解数据库压力,加快查询速度。
    • mybatis一级缓存是一个SqlSession级别,sqlsession只能访问自己的一级缓存的数据
    • 二级缓存是跨sqlSession,是mapper级别的缓存,对于mapper级别的缓存不同的sqlsession是可以共享的。

    一级查询缓存

    一级查询存在于每一个 SqlSession 类的实例对象中,当第一次查询某一个数据时,SqlSession 类的实例对象会将该数据存入一级缓存区域,在没有收到改变该数据的请求之前,用户再次查询该数据,都会从缓存中获取该数据,而不是再次连接数据库进行查询。

    • MyBatis 的一级缓存原理:

    MyBatis学习总结_第19张图片

    第一次发出一个查询 sql,sql 查询结果写入 sqlsession 的一级缓存中,缓存使用的数据结构是一个 map

    • key:hashcode+sql+sql输入参数+输出参数(sql的唯一标识)
    • value:用户信息

    同一个 sqlsession 再次发出相同的 sql,就从缓存中取不走数据库。如果两次中间出现 commit 操作(修改、添加、删除),本 sqlsession 中的一级缓存区域全部清空,下次再去缓存中查询不到所以要从数据库查询,从数据库查询到再写入缓存。

    一级缓存示例

    • 我们在同一个 session 中查询两次 id = 1 的 Category 对象试一试:
    public static void main(String[] args) throws IOException {
        String resource = "mybatis-config.xml";
        InputStream inputStream = Resources.getResourceAsStream(resource);
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        SqlSession session1 = sqlSessionFactory.openSession();
    
        Category c1 = session1.selectOne("getCategory", 1);
        System.out.println(c1);
        Category c2 = session1.selectOne("getCategory", 1);
        System.out.println(c2);
    
        session1.commit();
        session1.close();
    
    }

     

     

    运行,可以看到第一次会去数据库中取数据,但是第二次就不会访问数据库了,而是直接从session中取出来:

    MyBatis学习总结_第20张图片

    • 我们再来测试一下在不同 session 里查询相同的 id 数据
    public static void main(String[] args) throws IOException {
        String resource = "mybatis-config.xml";
        InputStream inputStream = Resources.getResourceAsStream(resource);
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        SqlSession session1 = sqlSessionFactory.openSession();
    
        Category c1 = session1.selectOne("getCategory", 1);
        System.out.println(c1);
        Category c2 = session1.selectOne("getCategory", 1);
        System.out.println(c2);
    
        session1.commit();
        session1.close();
    
        SqlSession session2 = sqlSessionFactory.openSession();
        Category c3 = session2.selectOne("getCategory", 1);
        System.out.println(c3);
        session2.commit();
        session2.close();
    
    }

     

    这一次,另外打开一个 session , 取同样 id 的数据,就会发现需要执行 sql 语句,证实了一级缓存是在 session 里的:

    MyBatis学习总结_第21张图片

    MyBatis 一级缓存值得注意的地方:

    • MyBatis 默认就是支持一级缓存的,并不需要我们配置.
    • MyBatis 和 spring 整合后进行 mapper 代理开发,不支持一级缓存,mybatis和 spring 整合,spring 按照 mapper 的模板去生成 mapper 代理对象,模板中在最后统一关闭 sqlsession。

    二级查询缓存

    • 问题: 有些时候,在 Web 工程中会将执行查询操作的方法封装在某个 Service 方法中,当查询完一次后,Service 方法结束,此时 SqlSession 类的实例对象就会关闭,一级缓存就会被清空。

    • 二级缓存原理:

    MyBatis学习总结_第22张图片

    二级缓存的范围是 mapper 级别(mapper即同一个命名空间),mapper 以命名空间为单位创建缓存数据结构,结构是 map。

    要开启二级缓存,需要进行两步操作。

    第一步:在 MyBatis 的全局配置文件 mybatis-config.xml 中配置 setting 属性,设置名为 “cacheEnable” 的属性值为 “true” 即可:

    
        
        
    

     

    • 注意: settings 配置的位置一定是在 properties 后面,typeAliases前面!

    第二步:然后由于二级缓存是 Mapper 级别的,还要在需要开启二级缓存的具体 mapper.xml 文件中开启二级缓存,只需要在相应的 mapper.xml 中添加一个 cache 标签即可:

    
    

     

    开启二级缓存之后,我们需要为查询结果映射的 POJO 类实现 java.io.serializable 接口,二级缓存可以将内存的数据写到磁盘,存在对象的序列化和反序列化,所以要实现java.io.serializable接口。

    二级缓存示例

    我们在同一个 SessionFactory 下查询 id = 1 的数据,只有第一次需要执行 SQL 语句,从后都是从缓存中取出来的:

    MyBatis学习总结_第23张图片

    参考资料:how2j.cn-MyBatis教程、Java3y-Mybatis【缓存、代理、逆向工程】

    参考资料:

    • 《Java EE 互联网轻量级框架整合开发》
    • 《Spring MVC + MyBatis开发从入门到项目实战》
    • How2j-MyBatis 系列教程

     

     

    转载自:“我没有三颗心脏”的两篇文章(微改)

    https://www.cnblogs.com/wmyskxz/p/8877109.html

    https://www.cnblogs.com/wmyskxz/p/8853461.html

     

你可能感兴趣的:(MyBatis学习总结)