一、MyBatis简介

  1. MyBatis历史

    MyBatis最初是Apache的一个开源项目iBatis, 2010年6月这个项目由Apache Software Foundation迁 移到Google Code。随着开发团队转投Google Code旗下, iBatis3.x正式更名为MyBatis。
    代码于 2013年11月迁移到Github。 iBatis一词来源于“internet”和“abatis”的组合,是一个基于Java的持久层框架。 iBatis提供的持久层框架 包括SQL Maps和Data Access Objects(DAO)。

  2. MyBatis特性

    1) MyBatis 是支持定制化 SQL、存储过程以及高级映射的优秀的持久层框架
    2) MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集
    3) MyBatis可以使用简单的XML或注解用于配置和原始映射,将接口和Java的POJO(Plain Old Java Objects,普通的Java对象)映射成数据库中的记录
    4) MyBatis 是一个 半自动的ORM(Object Relation Mapping)框架

  3. MyBatis下载

https://github.com/mybatis/mybatis-3
image.png

二、搭建MyBatis

  1. 开发环境

IDE:IntelliJ IDEA 2020.2.4 (Ultimate Edition)
构建工具:apache-maven 3.6.3
MySQL版本:8.0.26(远程服务器)
MyBatis版本:3.5.7

  1. 创建maven工程

    1. 打包方式:jar
    2. 引入依赖 ```properties
    org.mybatis mybatis 3.5.7 junit junit 4.12 test mysql mysql-connector-java 8.0.26

  1. <br />
  2. 3. **创建MyBatis核心配置文件**
  3. ![image.png](https://cdn.nlark.com/yuque/0/2022/png/28229185/1651660530377-0fa2090e-b397-46da-959c-b8300a008b2a.png#clientId=ua6177b53-cc8e-4&crop=0&crop=0&crop=1&crop=1&from=paste&height=329&id=ude2fe65b&margin=%5Bobject%20Object%5D&name=image.png&originHeight=1030&originWidth=1920&originalType=binary&ratio=1&rotation=0&showTitle=false&size=210778&status=done&style=none&taskId=ub1626db1-6368-4b31-87ee-5ed2bac519e&title=&width=614)
  4. ```xml
  5. <?xml version="1.0" encoding="UTF-8" ?>
  6. <!DOCTYPE configuration
  7. PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  8. "http://mybatis.org/dtd/mybatis-3-config.dtd">
  9. <configuration>
  10. <!-- 配置数据库连接环境-->
  11. <environments default="development">
  12. <environment id="development">
  13. <transactionManager type="JDBC"/>
  14. <dataSource type="POOLED">
  15. <property name="driver" value="com.mysql.cj.jdbc.Driver"/>
  16. <property name="url" value="jdbc:mysql://zhouge.xyz:3306/db_mybatis"/>
  17. <property name="username" value="admin"/>
  18. <property name="password" value="991227zg@."/>
  19. </dataSource>
  20. </environment>
  21. </environments>
  22. <!--引入映射文件-->
  23. <mappers>
  24. <mapper resource="mappers/UserMapper.xml"/>
  25. </mappers>
  26. </configuration>


  1. 创建mapper接口

image.png

  1. public interface UserMapper {
  2. int insertUser();
  3. }


  1. 创建MyBatis映射文件

image.png

  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE mapper
  3. PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
  4. "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <!--
  6. 两个一致:
  7. 1、namespace需要和映射文件的所在的全类名保持一致
  8. 2、id需要和执行的方法名保持一致
  9. -->
  10. <mapper namespace="xyz.zhouge.mybatis.mapper.UserMapper">
  11. <insert id="insertUser">
  12. insert into tb_user values(null,'喻雅婷1','123456789',20,'女','1974388746@qq.com')
  13. </insert>
  14. </mapper>


  1. 加入junit测试功能

image.png

  1. @Test
  2. public void testMyBatis(){
  3. SqlSession sqlSession = null ;
  4. try {
  5. //1、加载核心配置文件
  6. InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
  7. //2、获取 SqlSessionFactoryBuilder
  8. SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();
  9. //3、获取 SqlSessionFactory
  10. SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(is);
  11. //4、获取SqlSession
  12. sqlSession = sqlSessionFactory.openSession(); //ctrl + p openSession 默认不自动提交事务
  13. //5、获取Mapper对象
  14. UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
  15. //6、调用方法
  16. int res = userMapper.insertUser();
  17. if(res > 0){
  18. System.out.println("添加成功!");
  19. //7、 提交事务
  20. sqlSession.commit();
  21. }else{
  22. System.out.println("添加失败!");
  23. //8、 失败则回滚事务
  24. sqlSession.rollback();
  25. }
  26. } catch (IOException e) {
  27. e.printStackTrace();
  28. }finally {
  29. if(sqlSession != null){
  30. //9、 关闭连接会话
  31. sqlSession.close();
  32. }
  33. }
  34. }
  1. 加入log4j日志功能

    1. 引入依赖

      1. <!-- log4j日志 -->
      2. <dependency>
      3. <groupId>log4j</groupId>
      4. <artifactId>log4j</artifactId>
      5. <version>1.2.17</version>
      6. </dependency>
    2. 加入log4j的配置文件

image.png

  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE log4j:configuration SYSTEM "log4j.dtd">
  3. <log4j:configuration xmlns:log4j="http://jakarta.apache.org/log4j/">
  4. <appender name="STDOUT" class="org.apache.log4j.ConsoleAppender">
  5. <param name="Encoding" value="UTF-8" />
  6. <layout class="org.apache.log4j.PatternLayout">
  7. <param name="ConversionPattern" value="%-5p %d{MM-dd HH:mm:ss,SSS} %m (%F:%L) \n" />
  8. </layout>
  9. </appender>
  10. <logger name="java.sql">
  11. <level value="debug" />
  12. </logger>
  13. <logger name="org.apache.ibatis">
  14. <level value="info" />
  15. </logger>
  16. <root>
  17. <level value="debug" />
  18. <appender-ref ref="STDOUT" />
  19. </root>
  20. </log4j:configuration>

三、核心配置文件详解

  1. environment

    1. transactionManager:事务管理器
      1. type 可取值 JDBC 或 MANAGED
        1. jdbc 意味着事物的提交与回滚需要手动配置
        2. manged 意味着事务的提交与回滚交给其他的管理器来管理
    2. 数据源
      1. type 可取值POOLED UNPOOLED JNDI
        1. POOLED 意味着 数据的连接使用数据连接池技术
        2. UNPOOLED 意味着 不适用数据连接池技术
        3. JNDI 意味着 使用上下文中的数据源
          1. <environment id="test">
          2. <transactionManager type="JDBC"/>
          3. <dataSource type="POOLED">
          4. <property name="driver" value="${jdbc.Driver}"/>
          5. <property name="url" value="${jdbc.url}"/>
          6. <property name="username" value="${jdbc.username}"/>
          7. <property name="password" value="${jdbc.password}"/>
          8. </dataSource>
          9. </environment>
  2. properties

注意:需要在配置文件中 使用properties 标签引入jdbc.properties配置文件
image.png

  1. jdbc.Driver=com.mysql.cj.jdbc.Driver
  2. jdbc.url=jdbc:mysql://zhouge.xyz:3306/db_mybatis
  3. jdbc.username=admin
  4. jdbc.password=991227zg@.
  1. typeAliases

image.png

  1. <!--设置类型别名 type必须写,alias可写可不写且不区分大小写-->
  2. <typeAliases>
  3. <typeAlias type="xyz.zhouge.mybatis.bean.User" alias="User"/>
  4. </typeAliases>
  1. mappers

image.png

  1. <!--引入映射文件-->
  2. <mappers>
  3. <!--<mapper resource="mappers/UserMapper.xml"/>-->
  4. <!--
  5. 以包的方式引入mapper映射文件,要求如下:
  6. 1、包路径一致
  7. 2、mapper接口要和映射文件的名字一致
  8. -->
  9. <package name="xyz.zhouge.mybatis.mapper"/>
  10. </mappers>

四、MyBatis的增删改查

  1. 添加数据

    1. @Test
    2. public void testCURD(){
    3. SqlSession sqlSession = null;
    4. try {
    5. InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
    6. SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();
    7. SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(is);
    8. sqlSession = sqlSessionFactory.openSession(true);
    9. UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
    10. int res = userMapper.insertUser();
    11. //int res = userMapper.updateUser();
    12. //int res = userMapper.deleteUserById();
    13. if (res > 0){
    14. System.out.println("执行成功");
    15. }else{
    16. System.out.println("执行失败");
    17. }
    18. } catch (IOException e) {
    19. if(sqlSession!=null){
    20. sqlSession.rollback();
    21. }
    22. e.printStackTrace();
    23. }finally {
    24. if(sqlSession!=null){
    25. sqlSession.close();
    26. }
    27. }
    28. }
  2. 修改数据

    1. @Test
    2. public void testCURD(){
    3. SqlSession sqlSession = null;
    4. try {
    5. InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
    6. SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();
    7. SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(is);
    8. sqlSession = sqlSessionFactory.openSession(true);
    9. UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
    10. //int res = userMapper.insertUser();
    11. int res = userMapper.updateUser();
    12. //int res = userMapper.deleteUserById();
    13. if (res > 0){
    14. System.out.println("执行成功");
    15. }else{
    16. System.out.println("执行失败");
    17. }
    18. } catch (IOException e) {
    19. if(sqlSession!=null){
    20. sqlSession.rollback();
    21. }
    22. e.printStackTrace();
    23. }finally {
    24. if(sqlSession!=null){
    25. sqlSession.close();
    26. }
    27. }
    28. }
  3. 删除数据

    1. @Test
    2. public void testCURD(){
    3. SqlSession sqlSession = null;
    4. try {
    5. InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
    6. SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();
    7. SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(is);
    8. sqlSession = sqlSessionFactory.openSession(true);
    9. UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
    10. //int res = userMapper.insertUser();
    11. //int res = userMapper.updateUser();
    12. int res = userMapper.deleteUserById();
    13. if (res > 0){
    14. System.out.println("执行成功");
    15. }else{
    16. System.out.println("执行失败");
    17. }
    18. } catch (IOException e) {
    19. if(sqlSession!=null){
    20. sqlSession.rollback();
    21. }
    22. e.printStackTrace();
    23. }finally {
    24. if(sqlSession!=null){
    25. sqlSession.close();
    26. }
    27. }
    28. }
  4. 查询数据

    1. @Test
    2. public void testSelect(){
    3. SqlSession sqlSession = null ;
    4. try {
    5. InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
    6. SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();
    7. SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(is);
    8. sqlSession = sqlSessionFactory.openSession(true);
    9. UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
    10. //User user = userMapper.findUserById();
    11. //System.out.println(user);
    12. List<User> users = userMapper.findAllUser();
    13. users.forEach(user -> System.out.println(user));
    14. } catch (IOException e) {
    15. e.printStackTrace();
    16. }finally {
    17. if(sqlSession!=null){
    18. sqlSession.close();
    19. }
    20. }
    21. }
  5. 封装SqlSession

    1. public static SqlSession getSqlSession(){
    2. SqlSession sqlSession = null;
    3. try {
    4. InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
    5. SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(is);
    6. sqlSession = sqlSessionFactory.openSession(true);
    7. } catch (IOException e) {
    8. e.printStackTrace();
    9. }
    10. return sqlSession;
    11. }

五、MyBatis获取参数值的两种方式(重点)

MyBatis获取参数值的两种方式:${ }#{ } 。${ } 本质是字符串拼接 ; #{ } 本质是占位符赋值
${}使用字符串拼接的方式拼接sql,若为字符串类型或日期类型的字段进行赋值时,需要手动加单引 号;但是#{}使用占位符赋值的方式拼接sql,此时为字符串类型或日期类型的字段进行赋值时,可以自 动添加单引号

  1. 单个字面量类型的参数

    1. 添加mapper接口 ```java public interface ParameterMapper {

      List getAllUser();

      User getUserByUsername(String username);

}

  1. 2. 设置mapper映射
  2. ```java
  3. <!--User getUserByUsername(String username);-->
  4. <select id="getUserByUsername" resultType="user">
  5. <!--select * from tb_user where username = #{username}-->
  6. select * from tb_user where username = '${username}'
  7. </select>
  1. 设置传值方式
  2. 测试
    1. @Test
    2. public void testGetUserByUsername(){
    3. ParameterMapper mapper = SqlSessionUtil.getSqlSession().getMapper(ParameterMapper.class);
    4. User user = mapper.getUserByUsername("喻雅婷");
    5. System.out.println(user);
    6. }
  1. 多个字面量类型的参数

当mapper接口方法的参数为多个时,mybatis会将这些参数存放在一个map集合中,以两种方式进行存放:
以arg0,arg1,arg2…为键,以参数值为值进行存放
以param1,param2,param3…为键,以参数值为值进行存放

  1. <!--User checkLogin(String username , String password);-->
  2. <select id="checkLogin" resultType="user">
  3. <!--select * from tb_user where username = #{arg0} and password = #{arg1}-->
  4. select * from tb_user where username = '${param1}' and password = '${param2}';
  5. </select>
  1. @Test
  2. public void testCheckLogin(){
  3. ParameterMapper mapper = SqlSessionUtil.getSqlSession().getMapper(ParameterMapper.class);
  4. User user = mapper.checkLogin("喻雅婷","123456789");
  5. System.out.println(user);
  6. }
  1. map集合类型的参数

思路:将参数封装在map集合中,在配置文件里通过访问map的键的方式来获取值

  1. User checkLoginByMap(Map<String,Object> map);
  2. <!--User checkLoginByMap(String username , String password);-->
  3. <select id="checkLoginByMap" resultType="user">
  4. select * from tb_user where username = #{username} and password = #{password}
  5. </select>
  6. @Test
  7. public void testCheckLoginByMap(){
  8. ParameterMapper mapper = SqlSessionUtil.getSqlSession().getMapper(ParameterMapper.class);
  9. Map<String,Object> map = new HashMap<>();
  10. map.put("username","喻雅婷");
  11. map.put("password","123456789");
  12. //将参数封装在map集合中,在配置文件里通过访问map的键的方式来获取值
  13. User user = mapper.checkLoginByMap(map);
  14. System.out.println(user);
  15. }
  1. 实体类类型的参数 ```java int insertUser(User user);

!—int insertUser(User user);—> insert into tb_user values(null,#{username},#{password},#{age},#{sex},#{email});

@Test public void testInsertUser(){ ParameterMapper mapper = SqlSessionUtil.getSqlSession().getMapper(ParameterMapper.class); int res = mapper.insertUser(new User(null, “张三”, “123456”, 18, “男”, “zhangsan@163.com”)); System.out.println(res); }

  1. 5. 使用@Param标识参数
  2. ```java
  3. User checkLoginByParam(@Param("username") String username , @Param("password") String password);
  4. <!--User checkLoginByParam(@Param("username") String username ,@Param("password") String password);-->
  5. <select id="checkLoginByParam" resultType="user">
  6. select * from tb_user where username = #{username} and password = #{password};
  7. </select>
  8. @Test
  9. public void testCheckLoginByParam(){
  10. ParameterMapper mapper = SqlSessionUtil.getSqlSession().getMapper(ParameterMapper.class);
  11. User user = mapper.checkLoginByParam("admin", "123456");
  12. System.out.println(user);
  13. }

总结:获取参数两大类情况
1、@Param 注解标识
2、实体类获取

六、MyBatis的各种查询功能

MyBatis的各种查询功能:
1、若查询出的数据只有一条,
a>可以通过实体类对象接收
b>可以通过list集合接收
c>可以通过map集合接收
2、若查询出的数据有多条
a>可以通过list集合接收
b>可以通过map集合接收

  1. 查询一个实体类对象 ```java User getUserById(@Param(“id”) int id);
  1. <select id="getUserById" resultType="user">
  2. select * from tb_user where id = #{id};
  3. </select>

@Test public void testGetUserById(){ SelectMapper mapper = SqlSessionUtil.getSqlSession().getMapper(SelectMapper.class); User user = mapper.getUserById(12); System.out.println(user); }

  1. 2. 查询一个list集合
  2. ```java
  3. List<User> getAllUser();
  4. <!--List<User> getAllUser();-->
  5. <select id="getAllUser" resultType="user">
  6. select * from tb_user;
  7. </select>
  8. @Test
  9. public void testGetAllUser(){
  10. SelectMapper mapper = SqlSessionUtil.getSqlSession().getMapper(SelectMapper.class);
  11. List<User> users = mapper.getAllUser();
  12. users.forEach(user -> System.out.println(user));
  13. }
  1. 查询单个数据 ```java Integer getCount();
  1. <select id="getCount" resultType="java.lang.Integer">
  2. select count(*) from tb_user;
  3. </select>

@Test public void testGetCount(){ SelectMapper mapper = SqlSessionUtil.getSqlSession().getMapper(SelectMapper.class); Integer count = mapper.getCount(); System.out.println(count); }

  1. 4. 查询一条数据为map集合
  2. ```java
  3. Map<String,Object> getUserByIdToMap(@Param("id") Integer id);
  4. <!--Map<String,Object> getUserByIdToMap(@Param("id") Integer id);-->
  5. <select id="getUserByIdToMap" resultType="map">
  6. select * from tb_user where id = #{id};
  7. </select>
  8. @Test
  9. public void testGetUserByIdToMap(){
  10. SelectMapper mapper = SqlSessionUtil.getSqlSession().getMapper(SelectMapper.class);
  11. Map<String, Object> user = mapper.getUserByIdToMap(12);
  12. System.out.println(user);
  13. /*{password=123456, sex=男, id=12, age=22, email=1974388746@qq.com, username=admin}*/
  14. }
  1. 查询多条数据为map集合 ```java /List> getAllUserToMap();/

@MapKey(“id”) Map getAllUserToMap();

@Test public void testGetAllUserToMap(){ SelectMapper mapper = SqlSessionUtil.getSqlSession().getMapper(SelectMapper.class); //List> users = mapper.getAllUserToMap(); //users.forEach(user-> System.out.println(user));

  1. Map<String, Object> users = mapper.getAllUserToMap();
  2. System.out.println(users);

}

  1. <a name="oGcG7"></a>
  2. #### 七、特殊SQL的执行
  3. 1. 模糊查询
  4. ```java
  5. User getUserByUsername(@Param("username") String username);
  6. <!--User getUserByUsername(@Param("username") String username);-->
  7. <select id="getUserByUsername" resultType="user">
  8. <!--select * from tb_user where username like '%#username}%';-->
  9. <!--select * from tb_user where username like '%${username}%';-->
  10. select * from tb_user where username like "%"#{username}"%"
  11. </select>
  12. @Testc
  13. public void testGetUserByUsername(){
  14. SQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(SQLMapper.class);
  15. User user = mapper.getUserByUsername("admin");
  16. System.out.println(user);
  17. }
  1. 批量删除 ```java int deleteMore(@Param(“ids”) String ids);
  1. <delete id="deleteMore">
  2. delete from tb_user where id in (${ids})
  3. </delete>

@Test public void testDeleteMore(){ SQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(SQLMapper.class); int res = mapper.deleteMore(“11,23,50”); System.out.println(res); }

  1. 3. 动态设置表名
  2. ```java
  3. List<User> getAllUser(@Param("tableName") String tableName);
  4. <!--List<User> getAllUser(@Param("tableName") String tableName);-->
  5. <select id="getAllUser" resultType="user">
  6. select * from ${tableName};
  7. </select>
  8. @Test
  9. public void testGetAllUser(){
  10. SQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(SQLMapper.class);
  11. List<User> users = mapper.getAllUser("tb_user");
  12. users.forEach(user -> System.out.println(user));
  13. }
  1. 添加功能获取自增的主键 ```java int insertUser(User user);
  1. <insert id="insertUser" useGeneratedKeys="true" keyProperty="id">
  2. insert into tb_user values(null,#{username},#{password},#{age},#{sex},#{email});
  3. </insert>

@Test public void testInsertUser(){ SQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(SQLMapper.class); User user = new User(null,”诸葛”,”123456”,23,”男”,”12456@163.com”); mapper.insertUser(user); System.out.println(user); }t

  1. <a name="onrsz"></a>
  2. #### 八、自定义映射resultMap
  3. 场景:当数据库的字段名与javabean中的属性名不一致时,使用resultMap
  4. 1. resultMap处理字段和属性的映射关系
  5. 解决方案一:别名
  6. ```java
  7. /**
  8. * 获取所有员工-通过别名的方式解决问题
  9. * @return
  10. */
  11. List<Employee> getAllEmployee();
  12. <!--List<Employee> getAllEmployee();-->
  13. <select id="getAllEmployee" resultType="Employee">
  14. select emp_id as empId,emp_name as empName,emp_age as empAge,emp_sex as empSex,
  15. emp_email as empEmail,emp_depId as empDeptId from tb_emp;
  16. </select>
  17. @Test
  18. public void testGetAllEmployee(){
  19. EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class);
  20. List<Employee> employees = mapper.getAllEmployee();
  21. for (Employee employee : employees) {
  22. System.out.println(employee);
  23. }
  24. }

解决方案二:设置全局配置的mapUnderscoreToCamelCase

  1. /**
  2. * 获取所有员工-通过全局配置mapUnderscoreToCamelCase解决问题
  3. * @return
  4. */
  5. List<Employee> getAllEmployee2();
  6. <!--List<Employee> getAllEmployee2();-->
  7. <select id="getAllEmployee2" resultType="Employee">
  8. select * from tb_emp;
  9. </select>
  10. @Test
  11. public void testGetAllEmployee2(){
  12. EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class);
  13. List<Employee> employees = mapper.getAllEmployee2();
  14. for (Employee employee : employees) {
  15. System.out.println(employee);
  16. }
  17. }mapUnderscoreToCamelCase

解决方案三:在映射文件中设置resultMap

  1. /**
  2. * 获取所有员工-通过ResultMap解决问题
  3. * @return
  4. */
  5. List<Employee> getAllEmployee3();
  6. <!--
  7. resultMap :当数据库的字段名与javabean中的属性名不一致时,使用resultMap
  8. id:识别id curd中的唯一标识
  9. type:映射的类型
  10. id:数据库表中的主键
  11. column 数据库中的字段名
  12. property javabean中的属性名
  13. result:普通字段
  14. column 数据库中的字段名
  15. property javabean中的属性名
  16. -->
  17. <resultMap id="employeeResultMap" type="xyz.zhouge.mybatis.bean.Employee">
  18. <id column="emp_id" property="empId"/>
  19. <result column="emp_name" property="empName"/>
  20. <result column="emp_age" property="empAge"/>
  21. <result column="emp_sex" property="empSex"/>
  22. <result column="emp_email" property="empEmail"/>
  23. <result column="emp_depId" property="empDeptId"/>
  24. </resultMap>
  25. <!--List<Employee> getAllEmployee3();-->
  26. <select id="getAllEmployee3" resultMap="employeeResultMap">
  27. select * from tb_emp;
  28. </select>
  29. @Test
  30. public void testGetAllEmployee3(){
  31. EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class);
  32. List<Employee> employees = mapper.getAllEmployee3();
  33. for (Employee employee : employees) {
  34. System.out.println(employee);
  35. }
  36. }
  1. 多对一映射解决方案

    1. 级联方式处理映射关系 ```java public class Employee {

      private Integer empId; private String empName; private Integer empAge; private String empSex; private String empEmail; private Integer empDeptId; private Department department;

      @Override public String toString() { return “Employee{“ +

      1. "empId=" + empId +
      2. ", empName='" + empName + '\'' +
      3. ", empAge=" + empAge +
      4. ", empSex='" + empSex + '\'' +
      5. ", empEmail='" + empEmail + '\'' +
      6. ", empDeptId=" + empDeptId +
      7. ", department=" + department +
      8. '}';

      }

      public Employee() { }

      public Employee(Integer empId, String empName, Integer empAge, String empSex, String empEmail, Integer empDeptId) { this.empId = empId; this.empName = empName; this.empAge = empAge; this.empSex = empSex; this.empEmail = empEmail; this.empDeptId = empDeptId; }

      public Employee(Integer empId, String empName, Integer empAge, String empSex, String empEmail, Integer empDeptId, Department department) { this.empId = empId; this.empName = empName; this.empAge = empAge; this.empSex = empSex; this.empEmail = empEmail; this.empDeptId = empDeptId; this.department = department; }

      public Integer getEmpId() { return empId; }

      public void setEmpId(Integer empId) { this.empId = empId; }

      public String getEmpName() { return empName; }

      public void setEmpName(String empName) { this.empName = empName; }

      public Integer getEmpAge() { return empAge; }

      public void setEmpAge(Integer empAge) { this.empAge = empAge; }

      public String getEmpSex() { return empSex; }

      public void setEmpSex(String empSex) { this.empSex = empSex; }

      public String getEmpEmail() { return empEmail; }

      public void setEmpEmail(String empEmail) { this.empEmail = empEmail; }

      public Integer getEmpDeptId() { return empDeptId; }

      public void setEmpDeptId(Integer empDeptId) { this.empDeptId = empDeptId; }

      public Department getDepartment() { return department; }

      public void setDepartment(Department department) { this.department = department; }

}

Employee getEmployeeAndDepartment(@Param(“empId”) Integer empId);

@Test public void testGetEmployeeAndDepartment(){ EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class); Employee employee = mapper.getEmployeeAndDepartment(2); System.out.println(employee); }

  1. 2. 使用association处理映射关系
  2. ```java
  3. Employee getEmployeeAndDepartment2(@Param("empId") Integer empId);
  4. <resultMap id="getEmployeeAndDepartment2" type="xyz.zhouge.mybatis.bean.Employee">
  5. <id column="emp_id" property="empId"/>
  6. <result column="emp_name" property="empName"/>
  7. <result column="emp_age" property="empAge"/>
  8. <result column="emp_sex" property="empSex"/>
  9. <result column="emp_email" property="empEmail"/>
  10. <result column="emp_depId" property="empDeptId"/>
  11. <!-- 通过设置association属性解决多对一问题 -->
  12. <association property="department" javaType="xyz.zhouge.mybatis.bean.Department">
  13. <id column="dep_id" property="deptId"/>
  14. <result column="dep_name" property="deptName"/>
  15. </association>
  16. </resultMap>
  17. <select id="getEmployeeAndDepartment2" resultMap="getEmployeeAndDepartment2">
  18. select * from tb_emp left join tb_dept on tb_emp.emp_depId = tb_dept.dep_id where tb_emp.emp_id = #{empId};
  19. </select>
  20. @Test
  21. public void testGetEmployeeAndDepartment2(){
  22. EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class);
  23. Employee employee = mapper.getEmployeeAndDepartment2(3);
  24. System.out.println(employee);
  25. }
  1. 分步查询(重难点) ```java /**
    • 通过分步查询获取员工信息
    • 第一步:查询员工表信息
    • @param empId
    • @return */ Employee getEmployeeAndDepartmentByStep1(@Param(“empId”) Integer empId);

public interface DepartmentMapper {

  1. /**
  2. * 通过分布查询 获取员工信息
  3. * 第二步:查询部门表
  4. *
  5. * 将第一步查询的部门id 放入到column中 当作第二步查询的条件
  6. 此处的 @Param("emp_depId") 是第一步查询出来的结果
  7. * @param deptId
  8. * @return
  9. */
  10. Department getEmployeeAndDepartmentByStep2(@Param("emp_depId") Integer deptId);

}

@Test public void testGetEmployeeAndDepartmentByStep1(){ EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class); Employee employee = mapper.getEmployeeAndDepartmentByStep1(2); System.out.println(employee); }

  1. 4. 延迟加载
  2. ```java
  3. <?xml version="1.0" encoding="UTF-8" ?>
  4. <!DOCTYPE configuration
  5. PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  6. "http://mybatis.org/dtd/mybatis-3-config.dtd">
  7. <configuration>
  8. <settings>
  9. <!--通过配置全局的设置,将sql的下划线字段名 转换为 javabean中符合驼峰规则的属性名-->
  10. <setting name="mapUnderscoreToCamelCase" value="false"/>
  11. <!-- 开启延迟加载 -->
  12. <setting name="lazyLoadingEnabled" value="true"/>
  13. <!-- 是否按需加载(搭配延迟加载一起使用) -->
  14. <!--<setting name="aggressiveLazyLoading" value="true"/>-->
  15. </settings>
  16. </configuration>
  17. /**
  18. * 通过分步查询获取员工信息
  19. * 第一步:查询员工表信息
  20. * @param empId
  21. * @return
  22. */
  23. Employee getEmployeeAndDepartmentByStep1(@Param("empId") Integer empId);
  24. <resultMap id="getEmployeeAndDepartmentByStep1ResultMap" type="xyz.zhouge.mybatis.bean.Employee">
  25. <id column="emp_id" property="empId"/>
  26. <result column="emp_name" property="empName"/>
  27. <result column="emp_age" property="empAge"/>
  28. <result column="emp_sex" property="empSex"/>
  29. <result column="emp_email" property="empEmail"/>
  30. <result column="emp_depId" property="empDeptId"/>
  31. <!--
  32. 分布查询:
  33. property:需要查询的javaBean的属性值
  34. select:第二部查询的步骤
  35. column:注意,这里并非是前面数据库中查询的字段名,而是第二步查询的条件。取值是当前步骤查询的字段名
  36. 即,把当前步骤中查询出来的某一字段名,当作后续步骤查询的条件。
  37. fetchType:搭配延迟加载一起使用,当延迟加载未启用时,默认eager,即实时加载
  38. lazy:延迟加载
  39. eager:实时加载
  40. -->
  41. <association property="department"
  42. select="xyz.zhouge.mybatis.mapper.DepartmentMapper.getEmployeeAndDepartmentByStep2"
  43. column="emp_depId"/>
  44. </resultMap>
  45. <select id="getEmployeeAndDepartmentByStep1" resultMap="getEmployeeAndDepartmentByStep1ResultMap">
  46. select * from tb_emp where emp_id = #{empId};
  47. </select>
  48. public interface DepartmentMapper {
  49. /**
  50. * 通过分布查询 获取员工信息
  51. * 第二步:查询部门表
  52. *
  53. * 将第一步查询的部门id 放入到column中 当作第二步查询的条件
  54. 此处的 @Param("emp_depId") 是第一步查询出来的结果
  55. * @param deptId
  56. * @return
  57. */
  58. Department getEmployeeAndDepartmentByStep2(@Param("emp_depId") Integer deptId);
  59. }
  60. <resultMap id="getEmployeeAndDepartmentByStep2ResultMap" type="xyz.zhouge.mybatis.bean.Department">
  61. <id column="dep_id" property="deptId"/>
  62. <result column="dep_name" property="deptName"/>
  63. </resultMap>
  64. <select id="getEmployeeAndDepartmentByStep2" resultMap="getEmployeeAndDepartmentByStep2ResultMap">
  65. select * from tb_dept where dep_id = #{emp_depId};
  66. </select>
  67. @Test
  68. public void testGetEmployeeAndDepartmentByStep1(){
  69. EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class);
  70. Employee employee = mapper.getEmployeeAndDepartmentByStep1(2);
  71. System.out.println(employee.getEmpEmail());
  72. System.out.println(employee.getDepartment());
  73. }
  1. 一对多映射处理
    1. collection ```java /**
  • 处理一对多关系-collection 查询部门信息
  • @param deptId
  • @return */ Department getDepartmentAndEmployee(@Param(“deptId”) Integer deptId);
@Test public void testGetDepartmentAndEmployee(){ DepartmentMapper mapper = SqlSessionUtil.getSqlSession().getMapper(DepartmentMapper.class); Department department = mapper.getDepartmentAndEmployee(2); System.out.println(department); } 2. 分步查询java /** 处理一对多关系 -分步查询 @param deptId @return / Department getDepartmentByStep1(@Param(“deptId”) Integer deptId); / 分步查询,将第一步查询出来的dep_id的值作为当前步骤的查询条件进行查询 @param depId @return / List getDepartmentAndEmployeeByStep2(@Param(“dep_id”) Integer depId); @Test public void testGetDepartmentByStep1(){ DepartmentMapper mapper = SqlSessionUtil.getSqlSession().getMapper(DepartmentMapper.class); Department department = mapper.getDepartmentByStep1(3); System.out.println(department.getDeptName()); } <a name="uvPj2"></a> #### 九、动态SQL 1. ifjava public interface DynamicSQLMapper { / 通过动态sql if 根据条件查询员工信息 @param employee @return / Employee getEmployeeByCondition(Employee employee); }

/**

  • @BelongsProject: MyBatis
  • @BelongsPackage: xyz.zhouge.mybatis.test
  • @CreateTime: 2022-05-07 14:42
  • @Description: TODO
  • @Version: 1.0
  • @Author:zhouge */ public class TestDynamicSQLMapper {

    @Test public void testGetEmployeeByCondition(){

    1. DynamicSQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(DynamicSQLMapper.class);
    2. Employee employee = mapper.getEmployeeByCondition(new Employee(null, "张三", 23, "男", "zhangsan@163.com", 2));
    3. System.out.println(employee);

    } } ```

  1. where ```java public interface DynamicSQLMapper {

    /**

    • 通过动态sql where 根据条件查询员工信息
    • @param employee
    • @return */ Employee getEmployeeByCondition2(Employee employee); }
  1. <select id="getEmployeeByCondition2" resultMap="Employee">
  2. select * from tb_emp
  3. <where>
  4. <if test="empName != null and empName != ''">
  5. and emp_name = #{empName}
  6. </if>
  7. <if test="empAge != null and empAge != ''">
  8. and emp_age = #{empAge}
  9. </if>
  10. <if test="empName != null and empName != ''">
  11. and emp_name = #{empName}
  12. </if>
  13. <if test="empSex != null and empSex != ''">
  14. and emp_sex = #{empSex}
  15. </if>
  16. </where>

/**

  • @BelongsProject: MyBatis
  • @BelongsPackage: xyz.zhouge.mybatis.test
  • @CreateTime: 2022-05-07 14:42
  • @Description: TODO
  • @Version: 1.0
  • @Author:zhouge */ public class TestDynamicSQLMapper {

    @Test public void testGetEmployeeByCondition2(){

    1. DynamicSQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(DynamicSQLMapper.class);
    2. Employee employee = mapper.getEmployeeByCondition(new Employee(null, "张三", 23, "男", "zhangsan@163.com", 2));
    3. System.out.println(employee);

    } } ```

  1. trim ```java public interface DynamicSQLMapper {

    /**

    • 通过动态sql trim 根据条件查询员工信息 prefix | suffix : 在trim标签的前面或后面添加指定内容 prefixOverrides | suffixOverrides : 在trim标签的前面或后面去掉指定的内容
    • @param employee
    • @return */ Employee getEmployeeByCondition3(Employee employee); }
  1. <select id="getEmployeeByCondition3" resultMap="Employee">
  2. select * from tb_emp
  3. <trim prefix="where" suffixOverrides="and|or" prefixOverrides="and">
  4. <if test="empName != null and empName != ''">
  5. emp_name = #{empName} and
  6. </if>
  7. <if test="empAge != null and empAge != ''">
  8. emp_age = #{empAge} and
  9. </if>
  10. <if test="empName != null and empName != ''">
  11. emp_name = #{empName}
  12. </if>
  13. <if test="empSex != null and empSex != ''">
  14. and emp_sex = #{empSex}
  15. </if>
  16. </trim>

/**

  • @BelongsProject: MyBatis
  • @BelongsPackage: xyz.zhouge.mybatis.test
  • @CreateTime: 2022-05-07 14:42
  • @Description: TODO
  • @Version: 1.0
  • @Author:zhouge */ public class TestDynamicSQLMapper {

    @Test public void testGetEmployeeByCondition2(){

    1. DynamicSQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(DynamicSQLMapper.class);
    2. Employee employee = mapper.getEmployeeByCondition(new Employee(null, "张三", 23, "男", "zhangsan@163.com", 2));
    3. System.out.println(employee);

    } } ```

  1. choose、when、otherwise ```java /**
    • 通过动态sql choose-when-otherwise 查询员工信息
    • @param employee
    • @return */ List getEmployeeByChoose(Employee employee);
  1. <select id="getEmployeeByChoose" resultMap="Employee">
  2. select * from tb_emp
  3. <where>
  4. <choose>
  5. <when test="empName != null and empName != ''">
  6. emp_name = #{empName}
  7. </when>
  8. <when test="empAge != null and empAge != ''">
  9. emp_age = #{empAge}
  10. </when>
  11. <when test="empSex != null and empSex != ''">
  12. emp_sex= #{empSex}
  13. </when>
  14. <when test="empEmail != null and empEmail != ''">
  15. emp_email = #{empEmail}
  16. </when>
  17. <otherwise>
  18. emp_id = 1
  19. </otherwise>
  20. </choose>
  21. </where>

@Test public void testGetEmployeeByChoose(){ DynamicSQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(DynamicSQLMapper.class); List employees = mapper.getEmployeeByChoose(new Employee(null, “”, 23, “男”, “zhangsan@163.com”, 2)); employees.forEach(employee -> System.out.println(employee)); }

  1. 5. foreach
  2. ```java
  3. /**
  4. * 通过动态sql foreach 删除或添加员工
  5. * @param eids
  6. * @return
  7. */
  8. int deleteMoreByArray(@Param("eids") Integer[] eids);
  9. <!--int deleteMoreByArray(@Param("eids") Integer[] eids);-->
  10. <delete id="deleteMoreByArray">
  11. <!--
  12. delete from tb_emp where emp_id in
  13. <foreach collection="eids" item="eid" separator="," open="(" close=")">
  14. #{eid}
  15. </foreach>
  16. -->
  17. delete from tb_emp where
  18. <foreach collection="eids" item="eid" separator="or">
  19. emp_id = #{eid}
  20. </foreach>
  21. </delete>
  22. @Test
  23. public void testDeleteMoreByArray(){
  24. DynamicSQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(DynamicSQLMapper.class);
  25. int res = mapper.deleteMoreByArray(new Integer[]{14,15,16});
  26. System.out.println(res);
  27. }
  1. /**
  2. * 通过动态sql foreach 添加员工
  3. * @param employeeList
  4. * @return
  5. */
  6. int insertMoreByArray(@Param("employees") List<Employee> employeeList);
  7. <!--int insertMoreByArray(@Param("employees") List<Employee> employeeList);-->
  8. <insert id="insertMoreByArray">
  9. insert into tb_emp values
  10. <foreach collection="employees" item="employee" separator=",">
  11. (null,#{employee.empName},#{employee.empAge},#{employee.empSex},#{employee.empEmail},#{employee.empDeptId})
  12. </foreach>
  13. </insert>
  14. @Test
  15. public void testInsertMoreByArray(){
  16. DynamicSQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(DynamicSQLMapper.class);
  17. Employee emp1 = new Employee(null, "zg", 23, "男", "zg@163.com", 1);
  18. Employee emp2 = new Employee(null, "zg", 23, "男", "zg@163.com", 1);
  19. Employee emp3 = new Employee(null, "zg", 23, "男", "zg@163.com", 1);
  20. Employee emp4 = new Employee(null, "zg", 23, "男", "zg@163.com", 1);
  21. Employee emp5 = new Employee(null, "zg", 23, "男", "zg@163.com", 1);
  22. Employee emp6 = new Employee(null, "zg", 23, "男", "zg@163.com", 1);
  23. List<Employee> list = Arrays.asList(emp1, emp2, emp3, emp4, emp5, emp6);
  24. System.out.println(mapper.insertMoreByArray(list));
  25. }
  1. sql片段 ```java Employee getEmployeeByCondition4(@Param(“id”) Integer id);
emp_id,emp_name,emp_age,emp_sex,emp_email,emp_depId @Test public void testGetEmployeeByCondition4(){ DynamicSQLMapper mapper = SqlSessionUtil.getSqlSession().getMapper(DynamicSQLMapper.class); Employee employee = mapper.getEmployeeByCondition4(1); System.out.println(employee); } <a name="h4jlp"></a> #### 十、MyBatis缓存 1. **MyBatis的一级缓存 : 基于SQLSession的**java public interface CacheMapper { /* 根据员工id查询员工信息 @param employeeId @return */ Employee getEmployeeById(@Param(“id”) Integer employeeId); } emp_id,emp_name,emp_age,emp_sex,emp_email,emp_depId

public class TestCacheMapper {

  1. /**
  2. * MyBatis 一级缓存
  3. * 基于SQLSession的
  4. */
  5. @Test
  6. public void testGetEmployeeById(){
  7. //同一个sqlsession创建
  8. CacheMapper mapper1 = SqlSessionUtil.getSqlSession().getMapper(CacheMapper.class);
  9. Employee employee1 = mapper1.getEmployeeById(1);
  10. System.out.println(employee1);
  11. Employee employee2 = mapper1.getEmployeeById(1);
  12. System.out.println(employee2);
  13. }
  14. @Test
  15. public void testGetEmployeeById2(){
  16. //通过同一个sqlsession创建 不同的mapper
  17. SqlSession sqlSession = SqlSessionUtil.getSqlSession();
  18. CacheMapper mapper1 = sqlSession.getMapper(CacheMapper.class);
  19. CacheMapper mapper2 = sqlSession.getMapper(CacheMapper.class);
  20. Employee employee1 = mapper1.getEmployeeById(1);
  21. System.out.println(employee1);
  22. Employee employee2 = mapper2.getEmployeeById(1);
  23. System.out.println(employee2);
  24. }
  25. @Test
  26. public void testGetEmployeeById3(){
  27. //不同的sqlsession创建不同的mapper
  28. CacheMapper mapper1 = SqlSessionUtil.getSqlSession().getMapper(CacheMapper.class);
  29. CacheMapper mapper2 = SqlSessionUtil.getSqlSession().getMapper(CacheMapper.class);
  30. Employee employee1 = mapper1.getEmployeeById(1);
  31. System.out.println(employee1);
  32. Employee employee2 = mapper2.getEmployeeById(1);
  33. System.out.println(employee2);
  34. }

}

  1. **一级缓存失效的四种情况:**
  2. 1. 不同的SQL session对应不同的一级缓存
  3. 1. 同一个SQL session,但是查询条件不同
  4. 1. 同一个SQL session,两次查询期间执行了任何一次增删改操作
  5. 1. 同一个SQL session,两次查询期间手动清空了缓存
  6. 2. **MyBatis的二级缓存 基于SqlSessionFactor**y
  7. 开启二级缓存:
  8. 1. 在核心配置文件中,设置全局配置属性cacheEnabled="true",默认为true,不需要设置
  9. ```java
  10. <settings>
  11. <!--通过配置全局的设置,将sql的下划线字段名 转换为 javabean中符合驼峰规则的属性名-->
  12. <setting name="mapUnderscoreToCamelCase" value="false"/>
  13. <!-- 开启延迟加载 -->
  14. <setting name="lazyLoadingEnabled" value="true"/>
  15. <!-- 是否按需加载(搭配延迟加载一起使用) -->
  16. <!--<setting name="aggressiveLazyLoading" value="true"/>-->
  17. <!-- 开启二级缓存 -->
  18. <setting name="cacheEnabled" value="true"/>
  19. </settings>
  1. 在映射文件中设置标签

    1. <!-- 开启二级缓存 -->
    2. <cache />
  2. 二级缓存必须在SqlSession关闭或提交之后有效

  3. 查询的数据所转换的实体类类型必须实现序列化的接口

    1. /**
    2. * 测试二级缓存 : 基于SQLSessionFactory
    3. */
    4. @Test
    5. public void testSecondCache(){
    6. try {
    7. //获取mybatis配置文件资源
    8. InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
    9. //创建SqlSessionFactory
    10. SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(is);
    11. //创建SqlSession 并设置自动提交
    12. SqlSession sqlSession1 = sqlSessionFactory.openSession(true);
    13. //创建mapper
    14. CacheMapper mapper1 = sqlSession1.getMapper(CacheMapper.class);
    15. //sqlSession1 执行操作并关闭
    16. Employee employee1 = mapper1.getEmployeeById(1);
    17. sqlSession1.close();
    18. //sqlSession2
    19. SqlSession sqlSession2 = sqlSessionFactory.openSession(true);
    20. CacheMapper mapper2 = sqlSession2.getMapper(CacheMapper.class);
    21. //sqlSession2 执行操作并关闭
    22. Employee employee2 = mapper2.getEmployeeById(1);
    23. sqlSession2.close();
    24. } catch (IOException e) {
    25. e.printStackTrace();
    26. }
    27. }

    注意: 使二级缓存失效的情况: 两次查询之间执行了任意的增删改,会使一级和二级缓存同时失效

  1. 二级缓存的相关配置

    在mapper配置文件中添加的cache标签可以设置一些属性:
    eviction属性:缓存回收策略
    LRU(Least Recently Used) – 最近最少使用的:移除最长时间不被使用的对象。
    FIFO(First in First out) – 先进先出:按对象进入缓存的顺序来移除它们。
    SOFT – 软引用:移除基于垃圾回收器状态和软引用规则的对象。
    WEAK – 弱引用:更积极地移除基于垃圾收集器状态和弱引用规则的对象。
    默认的是 LRU。
    flushInterval属性:刷新间隔,单位毫秒 默认情况是不设置,也就是没有刷新间隔,缓存仅仅调用语句时刷新
    size属性:引用数目,正整数 代表缓存最多可以存储多少个对象,太大容易导致内存溢出

readOnly属性:只读,true/false
true:只读缓存;会给所有调用者返回缓存对象的相同实例。因此这些对象不能被修改。这提供了 很重要的性能优势。
false:读写缓存;会返回缓存对象的拷贝(通过序列化)。这会慢一些,但是安全,因此默认是 false。

  1. MyBatis缓存查询的顺序

先查询二级缓存,因为二级缓存中可能会有其他程序已经查出来的数据,可以拿来直接使用。
如果二级缓存没有命中,再查询一级缓存
如果一级缓存也没有命中,则查询数据库
SqlSession关闭之后,一级缓存中的数据会写入二级缓存

  1. 整合第三方缓存EHCache(代替二级缓存,一级缓存没法代替)
    1. 添加依赖 ```xml org.mybatis.caches mybatis-ehcache 1.2.1
ch.qos.logback logback-classic 1.2.3 2. **各种jar包功能** | jar包名称 | 作用 | | --- | --- | | mybatis-ehcache | Mybatis和EHCache的整合包 | | ehcache | EHCache核心 | | slf4j-api | SLF4J日志门面包 | | logback-classic | 支持SLF4J门面接口的一个具体实现 | 3. **创建EHCache配置文件ehcache.xml**xml <?xml version=”1.0” encoding=”utf-8” ?> 4. **设置二级缓存的类型**xml 5. **加入logback日志(文件名为log-back.xml)**xml <?xml version=”1.0” encoding=”UTF-8”?> [%d{HH:mm:ss.SSS}] [%-5level] [%thread] [%logger] [%msg]%n utf8 6. **EHCache配置文件说明** ![image.png](https://cdn.nlark.com/yuque/0/2022/png/28229185/1652113494555-fd93a811-0f11-4b34-accc-c00e0038bddc.png#clientId=uce39accc-2997-4&crop=0&crop=0&crop=1&crop=1&from=paste&height=538&id=uf35289ee&margin=%5Bobject%20Object%5D&name=image.png&originHeight=673&originWidth=553&originalType=binary&ratio=1&rotation=0&showTitle=false&size=159477&status=done&style=none&taskId=u4f223507-c376-425c-84ab-c5b31a4e2a4&title=&width=442.4) <a name="o30ot"></a> #### 十一、MyBatis的逆向工程 1. 创建逆向工厂的步骤 1. 添加依赖和插件xml org.mybatis mybatis 3.5.7
  1. <!-- junit测试 -->
  2. <dependency>
  3. <groupId>junit</groupId>
  4. <artifactId>junit</artifactId>
  5. <version>4.12</version>
  6. <scope>test</scope>
  7. </dependency>
  8. <!-- MySQL驱动 -->
  9. <dependency>
  10. <groupId>mysql</groupId>
  11. <artifactId>mysql-connector-java</artifactId>
  12. <version>8.0.26</version>
  13. </dependency>
  14. <!-- log4j日志 -->
  15. <dependency>
  16. <groupId>log4j</groupId>
  17. <artifactId>log4j</artifactId>
  18. <version>1.2.17</version>
  19. </dependency>
  20. </dependencies>
  21. <!-- 控制Maven在构建过程中相关配置 -->
  22. <build>
  23. <!-- 构建过程中用到的插件 -->
  24. <plugins>
  25. <!-- 具体插件,逆向工程的操作是以构建过程中插件形式出现的 -->
  26. <plugin>
  27. <groupId>org.mybatis.generator</groupId>
  28. <artifactId>mybatis-generator-maven-plugin</artifactId>
  29. <version>1.3.0</version>
  30. <!-- 插件的依赖 -->
  31. <dependencies>
  32. <!-- 逆向工程的核心依赖 -->
  33. <dependency>
  34. <groupId>org.mybatis.generator</groupId>
  35. <artifactId>mybatis-generator-core</artifactId>
  36. <version>1.3.2</version>
  37. </dependency>
  38. <!-- 数据库连接池 -->
  39. <dependency>
  40. <groupId>com.mchange</groupId>
  41. <artifactId>c3p0</artifactId>
  42. <version>0.9.5.2</version>
  43. </dependency>
  44. <dependency>
  45. <groupId>com.alibaba</groupId>
  46. <artifactId>druid</artifactId>
  47. <version>1.2.8</version>
  48. </dependency>
  49. <!-- MySQL驱动 -->
  50. <dependency>
  51. <groupId>mysql</groupId>
  52. <artifactId>mysql-connector-java</artifactId>
  53. <version>8.0.26</version>
  54. </dependency>
  55. </dependencies>
  56. </plugin>
  57. </plugins>
  58. </build>
  1. 2. 创建MyBatis的核心配置文件
  2. ```xml
  3. <?xml version="1.0" encoding="UTF-8" ?>
  4. <!DOCTYPE configuration
  5. PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  6. "http://mybatis.org/dtd/mybatis-3-config.dtd">
  7. <configuration>
  8. <properties resource="jdbc.properties"/>
  9. <settings>
  10. <!--通过配置全局的设置,将sql的下划线字段名 转换为 javabean中符合驼峰规则的属性名-->
  11. <setting name="mapUnderscoreToCamelCase" value="false"/>
  12. <!-- 开启延迟加载 -->
  13. <setting name="lazyLoadingEnabled" value="true"/>
  14. <!-- 是否按需加载(搭配延迟加载一起使用) -->
  15. <!--<setting name="aggressiveLazyLoading" value="true"/>-->
  16. <!-- 开启二级缓存 -->
  17. <setting name="cacheEnabled" value="true"/>
  18. </settings>
  19. <typeAliases>
  20. <!-- 等逆向工程完成后再添加 -->
  21. <typeAlias type="xyz.zhouge.mybatis.bean.Department"/>
  22. <typeAlias type="xyz.zhouge.mybatis.bean.Employee"/>
  23. </typeAliases>
  24. <environments default="test">
  25. <environment id="development">
  26. <transactionManager type="JDBC"/>
  27. <dataSource type="POOLED">
  28. <property name="driver" value=""/>
  29. <property name="url" value=""/>
  30. <property name="username" value=""/>
  31. <property name="password" value=""/>
  32. </dataSource>
  33. </environment>
  34. <environment id="test">
  35. <transactionManager type="JDBC"/>
  36. <dataSource type="POOLED">
  37. <property name="driver" value="${jdbc.Driver}"/>
  38. <property name="url" value="${jdbc.url}"/>
  39. <property name="username" value="${jdbc.username}"/>
  40. <property name="password" value="${jdbc.password}"/>
  41. </dataSource>
  42. </environment>
  43. </environments>
  44. <mappers>
  45. <package name="xyz.zhouge.mybatis.mapper"/>
  46. </mappers>
  47. </configuration>
  1. 创建逆向工程的配置文件 ```xml <?xml version=”1.0” encoding=”UTF-8”?> <!DOCTYPE generatorConfiguration PUBLIC “-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN” “http://mybatis.org/dtd/mybatis-generator-config_1_0.dtd">

  1. 4. 执行MBG插件的generate目标
  2. 2. QBC查询
  3. ```java
  4. package xyz.zhouge.mybatis.test;
  5. import org.junit.Test;
  6. import xyz.zhouge.mybatis.bean.Employee;
  7. import xyz.zhouge.mybatis.bean.EmployeeExample;
  8. import xyz.zhouge.mybatis.mapper.EmployeeMapper;
  9. import xyz.zhouge.mybatis.utils.SqlSessionUtil;
  10. import java.util.List;
  11. /**
  12. * @BelongsProject: MyBatis
  13. * @BelongsPackage: xyz.zhouge.mybatis.test
  14. * @CreateTime: 2022-05-10 15:44
  15. * @Description: TODO
  16. * @Version: 1.0
  17. * @Author:zhouge
  18. */
  19. public class TestEmployeeMapper {
  20. @Test
  21. public void testSelectAll(){
  22. EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class);
  23. List<Employee> employees = mapper.selectByExample(null);
  24. employees.forEach(employee -> System.out.println(employee));
  25. }
  26. @Test
  27. public void testSelectByExample(){
  28. EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class);
  29. //创建查询条件 年龄大于20 且 小于25
  30. EmployeeExample employeeExample = new EmployeeExample();
  31. employeeExample.createCriteria().andEmpAgeGreaterThan(20).andEmpAgeLessThan(25);
  32. //输出
  33. List<Employee> employees = mapper.selectByExample(employeeExample);
  34. employees.forEach(employee -> System.out.println(employee));
  35. }
  36. }

十二、分页插件

  1. 分页插件使用步骤

    1. 添加依赖

      1. <dependency>
      2. <groupId>com.github.pagehelper</groupId>
      3. <artifactId>pagehelper</artifactId>
      4. <version>5.2.0</version>
      5. </dependency>
    2. 配置分页插件

      1. <!--在mybatis核心配置文件中进行设置-->
      2. <plugins>
      3. <!--设置分页插件-->
      4. <plugin interceptor="com.github.pagehelper.PageInterceptor"></plugin>
      5. </plugins>
  2. 分页插件的使用 ```java package test;

import com.github.pagehelper.Page; import com.github.pagehelper.PageHelper; import com.github.pagehelper.PageInfo; import org.junit.Test; import xyz.zhouge.mybatis.bean.Employee; import xyz.zhouge.mybatis.bean.EmployeeExample; import xyz.zhouge.mybatis.mapper.EmployeeMapper; import xyz.zhouge.mybatis.utils.SqlSessionUtil;

import java.util.List;

/**

  • @BelongsProject: MyBatis
  • @BelongsPackage: xyz.zhouge.mybatis.test
  • @CreateTime: 2022-05-10 15:44
  • @Description: TODO
  • @Version: 1.0
  • @Author:zhouge */ public class TestEmployeeMapper {

    @Test public void testSelectAll(){

    1. EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class);
    2. /**
    3. * 在查询之前开启分页 PageHelper.startPage(int pageNum, int pageSize)
    4. *
    5. * pageNum:当前页的页码
    6. * pageSize:每页显示的条数
    7. */
    8. Page<Object> page = PageHelper.startPage(1, 3);
    9. List<Employee> employees = mapper.selectByExample(null);
    10. /**
    11. * PageInfo<T> pageInfo = new PageInfo<>(List<T> list, int navigatePages)获取分页相关数据
    12. */
    13. PageInfo<Employee> info = new PageInfo<>(employees, 2);
    14. employees.forEach(employee -> System.out.println(employee));
    15. System.out.println(info);
    16. /**
    17. * PageInfo
    18. * {
    19. *
    20. * pageNum=1, pageSize=3, size=3, startRow=1, endRow=3, total=13, pages=5, list=Page{count=true, pageNum=1, pageSize=3, startRow=0, endRow=3, total=13, pages=5, reasonable=false, pageSizeZero=false
    21. *
    22. * }
    23. *
    24. * [
    25. * Employee{empId=1, empName='张三', empAge=23, empSex='男', empEmail='zhangsan@163.com', empDepid=1},
    26. * Employee{empId=2, empName='李四', empAge=18, empSex='女', empEmail='lisi@qq.com', empDepid=2},
    27. * Employee{empId=3, empName='王五', empAge=23, empSex='男', empEmail='wangwu@163.com', empDepid=1}
    28. * ],
    29. *
    30. * prePage=0, nextPage=2, isFirstPage=true, isLastPage=false, hasPreviousPage=false, hasNextPage=true, navigatePages=2, navigateFirstPage=1, navigateLastPage=2, navigatepageNums=[1, 2]}
    31. */

    }

    @Test public void testSelectByExample(){

    1. EmployeeMapper mapper = SqlSessionUtil.getSqlSession().getMapper(EmployeeMapper.class);
    2. //创建查询条件 年龄大于20 且 小于25
    3. EmployeeExample employeeExample = new EmployeeExample();
    4. employeeExample.createCriteria().andEmpAgeGreaterThan(20).andEmpAgeLessThan(25);
    5. //输出
    6. List<Employee> employees = mapper.selectByExample(employeeExample);
    7. employees.forEach(employee -> System.out.println(employee));

    }

}

```