Mybatis笔记

Mybatis

1. JDBC操作分析

// 数据库连接
Connection connection = null;
// 预处理对象
PreparedStatement preparedStatement = null;
// 查询结果集
ResultSet resultSet = null;
try {
    // 1.加载驱动
    Class.forName("com.mysql.jdbc.Driver");

    // 2.获取连接
    connection = DriverManager.getConnection("jdbc:mysql://luojie.site:33306/demo",
                                             "root",
                                             "123456");

    // 定义执行SQL
    String sql = "select user_id,user_no,user_name from users where user_name = ?";
    // 3.获取预处理statement,设置参数
    preparedStatement = connection.prepareStatement(sql);
    preparedStatement.setString(1, "MySQL");

    // 4.执行SQL,获取结果集
    resultSet = preparedStatement.executeQuery();

    // 5.遍历结果集,封装数据
    User user = null;
    while (resultSet.next()) {
        user = new User();
        int userId = resultSet.getInt("user_id");
        String userNo = resultSet.getString("user_no");
        String userName = resultSet.getString("user_name");
        user.setUserId(userId);
        user.setUserNo(userNo);
        user.setUserName(userName);
    }
    System.out.println(user);
} catch (Exception e) {
    e.printStackTrace();
} finally {
    if (null != connection) {
        try {
            connection.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    if (null != preparedStatement) {
        try {
            preparedStatement.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    if (null != resultSet) {
        try {
            resultSet.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
}

1、频繁创建、释放数据库连接,影响系统性能。

2、SQL硬编码、不易维护。

3、SQL参数硬编码,SQL参数异变,不易维护。

4、对获取数据结果集硬编码,SQL变更,获取数据结果集需要便跟,系统不易维护。

2. MyBatis 是什么?

2.1 对象/关系映射(ORM)

ORM 全称 Object Relation Mapping :表示对象-关系映射的缩写

ORM 完成面向对象的编程语言到关系数据库的映射。当ORM框架完成映射后,程序员既可以利用面向对象程序设计语言的简单易用性,又可以利用关系数据库的技术优势。ORM把关系数据库包装面向对象的模型。ORM框架是面向对象设计语言与关系数据库发展不同步时的中间解决方案。采用ORM框架后应用程序不在直接访问底层数据库,而是以面向对象的方式来操作持久化对象,而ORM框架则将这些面向对象操作转换为底层SQL操作。ORM框架实现的效果:把对持久化对象的保存、修改、删除等操作,转换为对数据库的操作。

2.2 Mybatis 简介?

MyBatis 是一款优秀的持久层框架,它支持自定义 SQL、存储过程以及高级映射。MyBatis 免除了几乎所有的 JDBC 代码以及设置参数和获取结果集的工作。MyBatis 可以通过简单的 XML 或注解来配置和映射原始类型、接口和 Java POJO(Plain Old Java Objects,普通老式 Java 对象)为数据库中的记录。

2.3 Mybatis历史

  1. 原是Apache的一个开源项目iBatis,2010年6月这个项目由Apache Software Foundation 迁移到了Google Code,随着开发团队转投Google Code旗下,iBatis3.x正式更名为Mybatis,代码与2013年11月迁移到Github。

    Mybatis 官网:https://mybatis.org/mybatis-3/zh/index.html
    MyBatis 源码:https://github.com/mybatis/mybatis-3

  2. IBatis一词源于 internet 和 abatis 的组合,是一个基于java的持久层框架。

2.4 Mybatis优势

    Mybatis 是一个半自动的持久层框架,对应开发人员而言,核心的sql还是需要自己优化,sql和java编码隔离,功能边界清晰,一个专注业务,一个专注数据。

分析图如下:

image-20200611091841470.png

3. Mybatis 使用与说明

3.1 Mybatis 简单使用

3.1.1 快速入门

Mybatis开发步骤

1)添加Mybatis坐标

    
        
            org.mybatis
            mybatis
            3.5.3
        
        
            mysql
            mysql-connector-java
            5.1.49
        
        
            junit
            junit
            4.12
            test
        
        
            org.projectlombok
            lombok
            1.18.12
            compile
        
    

2)创建user数据表

CREATE TABLE `user` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `username` varchar(225) DEFAULT NULL,
  `password` varchar(225) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;


INSERT INTO `demo`.`user` (`id`, `username`, `password`) VALUES ('1', '张三', '123456');
INSERT INTO `demo`.`user` (`id`, `username`, `password`) VALUES ('2', '李四', '456789');
INSERT INTO `demo`.`user` (`id`, `username`, `password`) VALUES ('3', '王五', '987654');

3)编写User实体类

@Data
public class User {
    private Integer id;
    private String username;
    private String password;
}

4)编写映射文件UserMapper.xml





5)编写核心文件SqlMapConfig.xml




    
        
            
            
                
                
                
                
            
        
    
    
        
    

6)编写测试类

    // 查询所有用户
    @Test
    public void testFindAll() throws IOException {

        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 对应操作
        
        // 关闭连接
        sqlSession.close();
    }

3.1.2 增删改查操作

3.1.2 增操作

1)编写UserMapper.xml


    
        insert into user (username,password) value(#{username},#{password})
    

2)插入User代码

    @Test
    public void testInsertUser() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        User user = new User();
        user.setUsername("测试");
        user.setPassword("123456");
        // 插入操作
        int insert = sqlSession.insert("com.ltgx.mapper.UserMapper.insertUser", user);
        sqlSession.commit();
        // 打印结果数据
        System.out.println(insert);
        // 关闭连接
        sqlSession.close();
    }

注意事项

新增使用insert标签

在映射文件中使用parameterType属性指定要插入的数据类型

sql语句中使用#{实体属性名}方式引入实体中的值

插入操作使用的API是sqlSession.insert("命名空间.id",对象)

插入操作涉及数据库数据变化,所以要使用sqlSession对象显示的提交事务,即sqlSession.commit()

3.1.2 删操作

1)编写UserMapper.xml


    
        delete from user where id = #{id}
    

2)删除User代码

    @Test
    public void testDeleteUser() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 操作
        int delete = sqlSession.delete("com.ltgx.mapper.UserMapper.deleteUser", 4);
        sqlSession.commit();
        // 打印结果数据
        System.out.println(delete);
        // 关闭连接
        sqlSession.close();
    }

注意事项

删除使用delete标签

在映射文件中使用parameterType属性指定要修改的数据类型

sql语句中使用#{任意字符串}方式引入传递的单个参数

删除操作使用的API是sqlSession.delete("命名空间.id",参数)

删除操作涉及数据库数据变化,所以要使用sqlSession对象显示的提交事务,即sqlSession.commit()

3.1.2 改操作

1)编写UserMapper.xml


    
        update user set username=#{username},password=#{password} where id =#{id}
    

2)修改User代码

    @Test
    public void testUpdateUser() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        User user = new User();
        user.setId(2);
        user.setUsername("测试");
        user.setPassword("123456");
        // 操作
        int update = sqlSession.update("com.ltgx.mapper.UserMapper.updateUser", user);
        sqlSession.commit();
        // 打印结果数据
        System.out.println(update);
        // 关闭连接
        sqlSession.close();
    }

注意事项

修改使用update标签

在映射文件中使用parameterType属性指定要修改的数据类型

sql语句中使用#{实体属性名}方式引入实体中的值

修改操作使用的API是sqlSession.insert("命名空间.id",对象)

修改操作涉及数据库数据变化,所以要使用sqlSession对象显示的提交事务,即sqlSession.commit()

3.1.2 查操作

1)编写UserMapper.xml


    

2)修改User代码

    @Test
    public void testFindAll() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 查询所有数据
        List userList = sqlSession.selectList("com.ltgx.mapper.UserMapper.findAll");
        // 打印结果数据
        System.out.println(userList);
        // 关闭连接
        sqlSession.close();
    }

注意事项

查询使用select标签

在映射文件中使用resultType属性指定要返回的数据类型

查询操作使用的API是sqlSession.selectXXX("命名空间.id")

3.1.3 使用方式

3.1.3.1 传统DAO方式

1)编写DAO接口

public interface UserDao {

    List findAll() throws IOException;

}

2)编写DAO接口实现

public class UserDaoImpl implements UserDao {

    @Override
    public List findAll() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 查询所有数据
        List userList = sqlSession.selectList("com.ltgx.mapper.UserMapper.findAll");
        // 打印结果数据
        System.out.println(userList);
        // 关闭连接
        sqlSession.close();
        return userList;
    }
}

3)测试方式

    @Test
    public void test() throws IOException {
        UserDao userDao = new UserDaoImpl();
        List all = userDao.findAll();
        System.out.println(all);
    }
3.1.3.2 代理方式

代理开发方式采用Mybatis的代理开发方式实现DAO层的开发

Mapper 接口开发方法只需要编写Mapper接口(相当于DAO的接口),由Mybatis框架根据接口定义创建接口的动态代理对象,代理对象的方法同上边DAO接口实现类的方法。

Mapper接口开发遵循以下规范:

1)Mapper.xml文件中的namespace与Mapper接口的全限定类名相同

2)Mapper接口方法与Mapper.xml中定义的每个statement的id相同

3)Mapper接口方法的输入参数类型和Mapper.xml中定义的每个SQL的parameterType的类型相同

4)Mapper接口方法的输出参数类型和Mapper.xml中定义的每个SQL的resultTYPE的类型相同

编写Mapper接口代码

public interface UserMapper {

    List findAll();

}

测试代码

    @Test
    public void testMapperFindAll() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 获取 Mapper
        UserMapper mapper = sqlSession.getMapper(UserMapper.class);
        // 查询所有数据
        List users = mapper.findAll();
        // 打印结果数据
        System.out.println(users);
        // 关闭连接
        sqlSession.close();
    }

3.2 配置文件

3.2.1 SqlMapConfig.xml

MyBatis 核心配置文件,其核心配置文件层级关系

[图片上传失败...(image-fd2856-1613610185688)]

下面为mybatis常用配置解析

3.2.1.1 environments 标签



    
    
        
        
            
            
            
            
                
                
                
                
                
            
        
    

事务管理器(transactionManager)类型有两种:

  • JDBC:这个配置就是直接使用了JDBC的提交和回滚设置,它依赖于从数据源得到连接来管理事务作用域
  • MANAGED :这个配置几乎没做什么。它从不提交或回滚一个连接,而是让容器来管理事务的整个生命周期(比如 JEE 应用服务器的上下文)。 默认情况下它会关闭连接。然而一些容器并不希望连接被关闭,因此需要将 closeConnection 属性设置为 false 来阻止默认的关闭行为。

数据源(dataSource)类型有三种:

  • UNPOOLED:这个数据源的实现只是每次请求时打开和关闭连接。
  • POOLED:这种数据源的实现利用“池”的概念将JDBC连接对象组织起来。
  • JNDI:这个数据源的实现是为了能在如EJB或应用服务器这类容器中使用,容器可以集中或在外部配置数据源,然后放置一个JNDI上下文的引用。
3.2.1.2 mapper 标签

该标签的作用是加载映射的,加载方式有如下几种:

  • 相对类路径的资源引用

    
      
      
      
    
    
  • 使用完全限定资源定位符(URL)

    
    
      
      
      
    
    
  • 使用映射器接口实现类的完全限定类名

    
    
      
      
      
    
    
  • 将包内的映射器接口实现全部注册为映射器

    
    
      
    
    
3.2.1.3 properties 标签

习惯将数据源的配置信息单独抽取成一个properties文件,该标签可以加载额外配置的properties文件

image-20200611110632054.png
3.2.1.4 typeAliases 标签

类型别名是为java类型设置的一个短的名字

image-20200611110941593.png

上面是我们自定义的别名,下面是mybatis框架已经为我们定义好的一些常用的类型的别名

别名 映射的类型
_byte byte
_long long
_short short
_int int
_integer int
_double double
_float float
_boolean boolean
string String
byte Byte
long Long
short Short
int Integer
integer Integer
double Double
float Float
boolean Boolean
date Date
decimal BigDecimal
bigdecimal BigDecimal
object Object
map Map
hashmap HashMap
list List
arraylist ArrayList
collection Collection
iterator Iterator
3.2.1.5 settings 标签

MyBatis 中极为重要的调整设置,它们会改变 MyBatis 的运行时行为

设置名 描述 有效值 默认值
cacheEnabled 全局性地开启或关闭所有映射器配置文件中已配置的任何缓存。 true|false true
mapUnderscoreToCamelCase 是否开启驼峰命名自动映射,即从经典数据库列名 A_COLUMN 映射到经典 Java 属性名 aColumn。 true|false false

3.2.2 mapper.xml

mapper.xml 对象与关系映射的具体文件





在mybatis中每条SQL的唯一标识通过【namespace.id】mapper的namespace命名空间名+“.”+id组成。

3.2.2.1 select 标签

select 映射查询语句,用于数据查询,从数据库取出数据。





    
        
        
        
    

    

    


同时 select 标签提供了很多属性来配置每条语句的行为细节


        select id,username,password from user
    

    
@Data
public class User {
    private Integer id;
    private String username;
    private String password;
}

resultMap:指定映射关系

resultType:使用 JavaBean 或 POJO 进行结果集映射,映射时需要查询的列明与字段名一致(mapUnderscoreToCamelCase-是否开启驼峰命名自动映射,即从经典数据库列名 A_COLUMN 映射到经典 Java 属性名 aColumn)

3.2.2.4 参数

{} 、${}

一般情况下使用 #{参数名} 进行参数传递。

一个参数的情况下,#{参数名} 参数名可以随便写,为了提高可维护性、可读性,最好与参数名一致。

多个参数的情况下,#{参数名} 需要与传入的参数名进行对应。如果传入的是 HashMap 的情况,参数名为HashMap的key。如果传入的参数为 POJO 对象,需要与 POJO 对象的字段名一致。

${} : 进行字符串替换

直接在 SQL 语句中直接插入一个不转义的字符串。 比如 ORDER BY 子句

ORDER BY ${columnName}
3.2.2.5 sql 标签

SQL中可将重复的SQL提取出来,使用时用incloud引用即可。比如:






    
        select * from user
    

    

    
    

3.2.2.6 动态sql

动态 SQL 是 MyBatis 的强大特性之一。

mybatis配置文件中,前门我们使用的sql都是比较简单的,有时候业务逻辑复杂时,我们的SQL是需要动态变化的,此时在前面介绍的就不能满足需求了。

3.2.2.6.1 if 标签
    

我们根据实体类user的不同取值,使用不同的SQL语句进行查询。比如

id 不为空时可以根据id来查询

select * from user where id = ?

username不为空时根据username来查询

select * from user where username = ?

id和username都不为空时来查询

select * from user where id = ? and username = ?
3.2.2.6.2 foreach 标签

动态 SQL 的另一个常见使用场景是对集合进行遍历(尤其是在构建 IN 条件语句的时候)


最终sql:

select * from user where id in (?,?,?)

collection:代表要遍历的集合元素,主要编写是不要写#{}

open:代表语句的开始部分

close:代表语句的结束部分

item:代表遍历集合的每个元素,生成时的变量

separator:代表分隔符

3.2.2.6.3 choose、when、otherwise

有时候,我们不想使用所有的条件,而只是想从多个条件中选择一个使用。针对这种情况,MyBatis 提供了 choose 元素,它有点像 Java 中的 switch 语句。


3.2.2.6.4 trim、where、set

where

where 元素只会在子元素返回任何内容的情况下才插入 “WHERE” 子句。而且,若子句的开头为 “AND” 或 “OR”,where 元素也会将它们去除。


set

用于动态更新语句的类似解决方案叫做 setset 元素可以用于动态包含需要更新的列,忽略其它不更新的列。


  update Author
    
      username=#{username},
      password=#{password},
      email=#{email},
      bio=#{bio},
    
  where id=#{id}

trim

自定义 trim 元素来定制 whereset 元素的功能,prefixOverrides 属性会忽略通过管道符分隔的文本序列(注意此例中的空格是必要的)。


  ...

set 元素会动态地在行首插入 SET 关键字,并会删掉额外的逗号(这些逗号是在使用条件语句给列赋值时引入的)


  ...

3.2.2.6.5 bind

bind 元素允许你在 OGNL 表达式以外创建一个变量,并将其绑定到当前的上下文。


3.3 复杂映射开发

一对一和一对多使用 用户与订单 作为案例

用户表和订单表的关系:一个用户可以有多个订单,一个订单只能有一个用户

image-20200611140712362.png
CREATE TABLE `user` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `username` varchar(225) DEFAULT NULL,
  `password` varchar(225) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;

INSERT INTO `demo`.`user` (`id`, `username`, `password`) VALUES ('1', '张三', '123456');
INSERT INTO `demo`.`user` (`id`, `username`, `password`) VALUES ('2', '李四', '456789');
INSERT INTO `demo`.`user` (`id`, `username`, `password`) VALUES ('3', '王五', '987654');

CREATE TABLE `orders` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `orderTime` datetime DEFAULT NULL,
  `money` decimal(10,2) DEFAULT NULL,
  `uid` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8mb4;
INSERT INTO `demo`.`orders` (`id`, `orderTime`, `money`, `uid`) VALUES ('1', '2020-06-10 14:10:59', '25.50', '1');
INSERT INTO `demo`.`orders` (`id`, `orderTime`, `money`, `uid`) VALUES ('2', '2020-06-10 14:11:08', '33.30', '1');
INSERT INTO `demo`.`orders` (`id`, `orderTime`, `money`, `uid`) VALUES ('3', '2020-06-04 14:11:22', '12.00', '2');
INSERT INTO `demo`.`orders` (`id`, `orderTime`, `money`, `uid`) VALUES ('4', '2020-06-02 14:11:34', '16.00', '3');

3.3.1 一对一查询

查询SQL:

select * from orders o,user u where o.uid=u.id; 

查询结果:

image-20200611141537989.png

编写用户实体:

@Data
public class User {
    private Integer id;
    private String username;
    private String password;
}

编写订单实体:

@Data
public class Orders {
    private Integer id;
    private LocalDateTime orderTime;
    private BigDecimal money;

    // 一个订单对应一个用户
    private User user;
}

编写订单Mapper:

public interface OrdersMapper {

    List findAll();
}

编写OrdersMapper.xml





    
        
        
        
        
        
        
    

    





    
        
        
        
        
            
            
            
        
    

    

测试结果:

    @Test
    public void testMapperFindAllOrders() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 获取 Mapper
        OrdersMapper mapper = sqlSession.getMapper(OrdersMapper.class);
        // 查询所有数据
        List orders = mapper.findAll();
        for (Orders order : orders) {
            System.out.println(order);
        }
        // 关闭连接
        sqlSession.close();
    }
image-20200611142739486.png

3.3.2 一对多查询

查询SQL:

select *,o.id oid from user u left join orders o on u.id=o.uid;

查询结果:

image-20200611143049477.png

编写用户实体:

@Data
public class User {
    private Integer id;
    private String username;
    private String password;

    // 一个用户有多个订单
    private List orders;
}

编写订单实体:

@Data
public class Orders {
    private Integer id;
    private LocalDateTime orderTime;
    private BigDecimal money;

    // 一个订单对应一个用户
    private User user;
}

编写订单Mapper:

public interface UserMapper {

    List findAll();

}

编写UserMapper.xml



    
        
        
        
        
            
            
            
        
    

    


测试结果:

    @Test
    public void testMapperFindAll() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 获取 Mapper
        UserMapper mapper = sqlSession.getMapper(UserMapper.class);
        // 查询所有数据
        List users = mapper.findAll();
        for (User user : users) {
            System.out.println(user.getUsername());
            for (Orders order : user.getOrders()) {
                System.out.println(order);
            }
            System.out.println("--------------------人工分割线-----------------");
        }
        // 关闭连接
        sqlSession.close();
    }
image-20200611144114240.png

3.3.3 多对多查询

多对多使用 用户与角色 作为案例

用户和角色关系:一个用户可以有多个角色,一个角色可以被多个人使用

image-20200611145143812.png
CREATE TABLE `role` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `rolename` varchar(255) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;

CREATE TABLE `user_role` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `user_id` int(11) DEFAULT NULL,
  `role_id` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf8mb4;

INSERT INTO `demo`.`role` (`id`, `rolename`) VALUES ('1', 'CTO');
INSERT INTO `demo`.`role` (`id`, `rolename`) VALUES ('2', 'CEO');
INSERT INTO `demo`.`role` (`id`, `rolename`) VALUES ('3', 'CXO');

INSERT INTO `demo`.`user_role` (`id`, `user_id`, `role_id`) VALUES ('1', '1', '1');
INSERT INTO `demo`.`user_role` (`id`, `user_id`, `role_id`) VALUES ('2', '1', '2');
INSERT INTO `demo`.`user_role` (`id`, `user_id`, `role_id`) VALUES ('3', '1', '3');
INSERT INTO `demo`.`user_role` (`id`, `user_id`, `role_id`) VALUES ('4', '2', '3');
INSERT INTO `demo`.`user_role` (`id`, `user_id`, `role_id`) VALUES ('5', '3', '2');

查询SQL语句:

SELECT
    u.*, r.*, r.id rid
FROM
    USER u
LEFT JOIN user_role ur ON u.id = ur.user_id
INNER JOIN role r ON ur.role_id = r.id;

创建Role实体类

@Data
public class Role {
    private Integer id;
    private String rolename;
}

User实体类

@Data
public class User {
    private Integer id;
    private String username;
    private String password;

    // 一个用户有多个订单
    private List orders;
    // 当前用户对应的角色
    private List roles;
}

编写UserMapper接口

public interface UserMapper {

    List findAllUserRole();
}

编写UserMapper.xml





    
        
        
        
        
            
            
            
        
        
            
            
        
    

    

测试结果

    @Test
    public void testMapperFindAllUserRole() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 获取 Mapper
        UserMapper mapper = sqlSession.getMapper(UserMapper.class);
        // 查询所有数据
        List users = mapper.findAllUserRole();
        for (User user : users) {
            System.out.println(user.getUsername());
            for (Role role : user.getRoles()) {
                System.out.println(role);
            }
            System.out.println("--------------------人工分割线-----------------");
        }
        // 关闭连接
        sqlSession.close();
    }
image-20200611150734952.png

3.4 注解开发

随着注解越来越流行,Mybatis也可以使用注解开发方式,这样我们就可以减少编写Mapper映射文件。

3.4.1 常用注解

注解 说明
@Select 实现查询
@Insert 实现新增
@Update 实现修改
@Delete 实现删除
@Result 实现结果集封装
@Results 可以与@Result结合使用,封装多个结果集
@One 实现一对一结果集封装
@Many 实现一对多结果集封装
3.4.1.1 @Results

@Results:代替的是标签,该注解中可以使用单个@Result注解,也可以使用@Result集合,使用格式:

@Results({
    @Result(),
    @Result()
})
@Results(@Result())

@Result:代替了标签,其@Result注解的属性:

column:数据库中的列表

property:需要装配的属性名

one:需要使用的@One注解,@Results({@Result(one = @One())})

many:需要使用的@Many注解,@Results({@Result(many = @Many())})

3.4.1.2 @One
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({})
public @interface One {
    String select() default "";

    FetchType fetchType() default FetchType.DEFAULT;
}

select为关联需要查询SQL的全局唯一标识ID【namespace.id】

3.4.1.3 @Many
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({})
public @interface Many {
    String select() default "";

    FetchType fetchType() default FetchType.DEFAULT;
}

select为关联需要查询SQL的全局唯一标识ID【namespace.id】

3.4.2 一对一

编写实体对象

@Data
public class Orders {
    private Integer id;
    private LocalDateTime orderTime;
    private BigDecimal money;

    // 一个订单对应一个用户
    private User user;
}

@Data
public class User {
    private Integer id;
    private String username;
    private String password;
}

编写mapper接口

public interface OrdersAnnotationMapper {

    @Select("select * from orders")
    @Results({
            @Result(column = "id", property = "id"),
            @Result(column = "orderTime", property = "orderTime"),
            @Result(column = "money", property = "money"),
            @Result(column = "uid", property = "user", javaType = User.class, one = @One(select = "com.ltgx.mapper.UserAnnotationMapper.selectUserById"))
    })
    List selectOrders();
}

public interface UserAnnotationMapper {

    @Select("select * from user where id = #{id}")
    User selectUserById(Integer id);

}

测试代码:

    @Test
    public void testOneToOne() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 获取 Mapper
        OrdersAnnotationMapper mapper = sqlSession.getMapper(OrdersAnnotationMapper.class);
        // 查询所有数据
        List ordersList = mapper.selectOrders();
        for (Orders orders : ordersList) {
            System.out.println(orders);
        }
        // 关闭连接
        sqlSession.close();
    }
image-20200611155043694.png

3.4.2 一对多

编写实体对象

@Data
public class Orders {
    private Integer id;
    private LocalDateTime orderTime;
    private BigDecimal money;

    // 一个订单对应一个用户
    private User user;
}

@Data
public class User {
    private Integer id;
    private String username;
    private String password;
    
    // 一个用户有多个订单
    private List orders;
}

编写mapper接口

public interface UserAnnotationMapper {

    @Select("select * from user")
    @Results({
            @Result(column = "id", property = "id"),
            @Result(column = "username", property = "username"),
            @Result(column = "password", property = "password"),
            @Result(property = "orders", column = "id", javaType = List.class, many = @Many(select = "com.ltgx.mapper.OrdersAnnotationMapper.selectOrdersByUserId"))
    })
    List selectUsers();
}

public interface OrdersAnnotationMapper {

    @Select("select * from orders where uid = #{userId}")
    List selectOrdersByUserId(Integer userId);

}

测试代码:

    @Test
    public void selectUsers() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 获取 Mapper
        UserAnnotationMapper mapper = sqlSession.getMapper(UserAnnotationMapper.class);
        // 查询所有数据
        List userList = mapper.selectUsers();
        for (User user : userList) {
            System.out.println(user.getUsername());
            for (Orders order : user.getOrders()) {
                System.out.println(order);
            }
            System.out.println("--------------------优雅的分隔符--------------------");
        }
        // 关闭连接
        sqlSession.close();
    }
image-20200611160827966.png

3.4.2 多对多

编写实体对象

@Data
public class Orders {
    private Integer id;
    private LocalDateTime orderTime;
    private BigDecimal money;

    // 一个订单对应一个用户
    private User user;
}

@Data
public class User {
    private Integer id;
    private String username;
    private String password;
    
    // 当前用户对应的角色
    private List roles;
}

编写mapper接口

public interface UserAnnotationMapper {

    @Select("select * from user")
    @Results({
            @Result(column = "id", property = "id"),
            @Result(column = "username", property = "username"),
            @Result(column = "password", property = "password"),
            @Result(property = "roles", column = "id", javaType = List.class, many = @Many(select = "com.ltgx.mapper.RoleMapper.selectRolesByUserId"))
    })
    List selectUsersAndRoles();
}

public interface RoleMapper {

    @Select("select r.id,r.rolename from role r,user_role ur where r.id=ur.role_id and ur.user_id=#{userId}")
    List selectRolesByUserId(Integer userId);

}

测试代码:

    @Test
    public void selectUsersAndRoles() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 获取 Mapper
        UserAnnotationMapper mapper = sqlSession.getMapper(UserAnnotationMapper.class);
        // 查询所有数据
        List userList = mapper.selectUsersAndRoles();
        for (User user : userList) {
            System.out.println(user.getUsername());
            for (Role role : user.getRoles()) {
                System.out.println(role);
            }
            System.out.println("--------------------优雅的分隔符--------------------");
        }
        // 关闭连接
        sqlSession.close();
    }
image-20200611161812056.png

4. Mybatis 拓展

4.1 Mybatis 插件

4.1.1 插件简介

    一般情况下,开源框架都还提供插件或其他形式的拓展点,供开发者自行拓展。这样的好处是显而易见的,一是增加了框架的灵活性,二是开发者可以结合实际需求,对框架进行拓展,使其更好的工作。

    以Mybatis为例,我们可以基于Mybatis插件机制实现分页、分表、监控等。由于业务和插件无关,业务也无法感知插件的存在。因此可以无感织入插件,在无形中增强功能。

4.1.2 Mybatis 插件介绍

    Mybatis是一个应用广泛的优秀ORM开源框架,这个框架具有强大的灵活性,在四大组件(Executor、StatementHandler、ParameterHandler、ResultSetHandler)出提供了简单易用的插件扩展机制。

    Mybatis对持久层的操作就是借助与四大核心对象。Mybatis支持用插件对四大核心组件进行拦截,对Mybatis来说插件就是拦截器,用来增强核心对象的功能,增强功能本质上是借助与底层的动态代理实现的,换句话说,Mybatis中的四大对象都是代理对象
image-20200611163546833.png

MyBatis 所允许拦截的方法如下:

  • 执行器(Executor):update,query,commit,rollback等

  • SQL语法构建器(StatementHandler):prepare,batch,update,query等

  • 参数处理器(ParameterHandler):getParameterObject,setParameters

  • 结果集处理器(ResultSetHandler):handleResultSets,handleCursorResultSets,handleOutputParameters

4.1.3 Mybatis 插件原理

四大对象创建后流程解析

1)每个创建出来的对象都不是直接返回的, 而是由interceptorChain.plugin(xxxHandler)生成的代理对象

2)InterceptorChain中获取到所有的interceptor(拦截器,插件需要实现的接口), 再调用interceptor.plugin(target),返回target包装后的对象

3)插件机制,我们可以使用插件为目标对象创建一个代理对象;类似AOP,我们的插件可以为四大组件创建出代理对象,代理对象就可以拦截到四大对象的每一个执行。

以 Executor 为例

public class DefaultSqlSessionFactory implements SqlSessionFactory {
......
  private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) {
    Transaction tx = null;
    try {
      final Environment environment = configuration.getEnvironment();
      final TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);
      tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);
      // 创建执行器
      final Executor executor = configuration.newExecutor(tx, execType);
      return new DefaultSqlSession(configuration, executor, autoCommit);
    } catch (Exception e) {
      closeTransaction(tx); // may have fetched a connection so lets call close()
      throw ExceptionFactory.wrapException("Error opening session.  Cause: " + e, e);
    } finally {
      ErrorContext.instance().reset();
    }
  }
    .......
}
public class Configuration {

......
  public Executor newExecutor(Transaction transaction, ExecutorType executorType) {
    executorType = executorType == null ? defaultExecutorType : executorType;
    executorType = executorType == null ? ExecutorType.SIMPLE : executorType;
    Executor executor;
    // 根据执行器类型构建执行器
    if (ExecutorType.BATCH == executorType) {
      executor = new BatchExecutor(this, transaction);
    } else if (ExecutorType.REUSE == executorType) {
      executor = new ReuseExecutor(this, transaction);
    } else {
      executor = new SimpleExecutor(this, transaction);
    }
    if (cacheEnabled) {
      executor = new CachingExecutor(executor);
    }
    // 构建执行器后通过 interceptorChain 生成代理对象
    executor = (Executor) interceptorChain.pluginAll(executor);
    return executor;
  }
......
}
public class InterceptorChain {

  private final List interceptors = new ArrayList<>();

  public Object pluginAll(Object target) {
    for (Interceptor interceptor : interceptors) {
      // 获取所有拦截器,创建代理类
      target = interceptor.plugin(target);
    }
    return target;
  }

  public void addInterceptor(Interceptor interceptor) {
    interceptors.add(interceptor);
  }

  public List getInterceptors() {
    return Collections.unmodifiableList(interceptors);
  }

}
public interface Interceptor {

  Object intercept(Invocation invocation) throws Throwable;

  default Object plugin(Object target) {
    return Plugin.wrap(target, this);
  }

  default void setProperties(Properties properties) {
    // NOP
  }

}
public class Plugin implements InvocationHandler {
......
  // 创建代理对象
  public static Object wrap(Object target, Interceptor interceptor) {
    Map, Set> signatureMap = getSignatureMap(interceptor);
    Class type = target.getClass();
    Class[] interfaces = getAllInterfaces(type, signatureMap);
    if (interfaces.length > 0) {
      return Proxy.newProxyInstance(
          type.getClassLoader(),
          interfaces,
          new Plugin(target, interceptor, signatureMap));
    }
    return target;
  }

  @Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    try {
        // 获取被拦截器拦截的方法列表
      Set methods = signatureMap.get(method.getDeclaringClass());
        // 检测方法列表是否包含在拦截方法
      if (methods != null && methods.contains(method)) {
          // 执行插件逻辑
        return interceptor.intercept(new Invocation(target, method, args));
      }
        // 执行被拦截的原方法
      return method.invoke(target, args);
    } catch (Exception e) {
      throw ExceptionUtil.unwrapThrowable(e);
    }
  }
......
}

4.1.4 自定义插件

1)插件类编码

/**
 * @author jie.luo
 * @description SQL执行之间统计
 * @since 2020-06-11
 */
@Intercepts({
        @Signature(
                type = Executor.class,
                method = "query",
                args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class}
        )
})
public class SqlExecutionTimeInterceptor implements Interceptor {

    /**
     * 拦截方法:只要被拦截的目标对象的方法被执行时,每次都会执行该方法
     */
    @Override
    public Object intercept(Invocation invocation) throws Throwable {

        System.out.println("开始时间 : " + LocalDateTime.now());
        // 让原方法执行
        Object proceed = invocation.proceed();
        System.out.println("结束时间 : " + LocalDateTime.now());
        return proceed;
    }

    /**
     * 把当前的拦截器,生成代理,存到拦截器链中
     *
     * @param target 被拦截的目标对象
     */
    @Override
    public Object plugin(Object target) {
        return Plugin.wrap(target, this);
    }

    /**
     * 获取配置文件的参数
     */
    @Override
    public void setProperties(Properties properties) {
        System.out.println("参数 " + properties);
    }
}

2)SQLMapConfig.xml配置插件


    
        
    

3)测试代码

@Test
public void selectOrders() throws IOException {
    // 加载核心配置文件
    InputStream resourceAsStream =
    Resources.getResourceAsStream("SqlMapConfig.xml");
    // 获取SqlSession 工厂
    SqlSessionFactory sqlSessionFactory = new
    SqlSessionFactoryBuilder().build(resourceAsStream);
    // 获取SqlSession 连接
    SqlSession sqlSession = sqlSessionFactory.openSession();
    // 获取 Mapper
    OrdersAnnotationMapper mapper = sqlSession.getMapper(OrdersAnnotationMapper.class);
    // 查询所有数据
    List ordersList = mapper.selectOrders();
    for (Orders orders : ordersList) {
    System.out.println(orders);
    }
    // 关闭连接
    sqlSession.close();
}
image-20200611174507476.png

4.2 分页插件

MyBatis 可以使用第三方的插件来对功能进行扩展,分页组手 pagehelper 是将分页的复杂操作进行封装,

使用简单的方式既可以获得分页相关的相关数据。

1)添加依赖


    com.github.pagehelper
    pagehelper
    5.1.11

2)核心配置文件添加分页插件


    

3)测试代码

    private UserAnnotationMapper userAnnotationMapper;

    @Before
    public void testBefore() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();
        // 获取 Mapper
        userAnnotationMapper = sqlSession.getMapper(UserAnnotationMapper.class);

    }

    @Test
    public void page() {

        PageHelper.startPage(1, 1);
        // 查询所有数据
        List userList = userAnnotationMapper.selectUsers();
        PageInfo pageInfo = new PageInfo<>(userList);
        
        // 分页相关参数
        System.out.println("总条数 " + pageInfo.getTotal());
        System.out.println("总页数 " + pageInfo.getPages());
        System.out.println("页码 " + pageInfo.getPageNum());
        System.out.println("页大小 " + pageInfo.getPageSize());
        System.out.println("是否为首页 " + pageInfo.isIsFirstPage());
        System.out.println("是否为末页 " + pageInfo.isIsLastPage());

        System.out.println(pageInfo);

    }

image-20200611180657128.png

4.3 通用mapper

通用Mapper就是为了解决单表增删改查,基于Mybatis插件机制。开发时不需要编写SQL,不需要在DAO增加方法,只要写好实体类,就能支持相应的增删改查方法。

通用Mapper Github地址:https://github.com/abel533/Mapper

1)导入依赖


    tk.mybatis
    mapper
    4.1.5

2)测试代码

public class TkMapperTest {

    private RoleMapper roleMapper;

    @Before
    public void testBefore() throws IOException {
        // 加载核心配置文件
        InputStream resourceAsStream =
                Resources.getResourceAsStream("SqlMapConfig.xml");
        // 获取SqlSession 工厂
        SqlSessionFactory sqlSessionFactory = new
                SqlSessionFactoryBuilder().build(resourceAsStream);
        // 获取SqlSession 连接
        SqlSession sqlSession = sqlSessionFactory.openSession();

        MapperHelper mapperHelper = new MapperHelper();
        mapperHelper.processConfiguration(sqlSession.getConfiguration());

        // 获取 Mapper
        roleMapper = sqlSession.getMapper(RoleMapper.class);
    }

    @Test
    public void test() {

        // 查询所有
        List roles = roleMapper.selectAll();
        System.out.println(roles);

        // Example 用法
        Example example = new Example(Role.class);
        Example.Criteria criteria = example.createCriteria();
        criteria.andLike("rolename", "%C%");
        example.orderBy("id").desc();
        List roles1 = roleMapper.selectByExample(example);
        System.out.println(roles1);

        // Example 用法,设置查询列
        Example example1 = new Example(Role.class);
        example1.selectProperties("rolename");
        Example.Criteria criteria1 = example1.createCriteria();
        criteria1.andLike("rolename", "%C%");
        List roles2 = roleMapper.selectByExample(example1);
        System.out.println(roles2);

    }

}

SQL

==>  Preparing: SELECT id,rolename FROM role 
==> Parameters: 
<==    Columns: id, rolename
<==      Total: 3
[Role(id=1, rolename=CTO), Role(id=2, rolename=CEO), Role(id=3, rolename=CXO)]

==>  Preparing: SELECT id,rolename FROM role WHERE ( ( rolename like ? ) ) order by id DESC 
==> Parameters: %C%(String)
<==    Columns: id, rolename
<==      Total: 3
[Role(id=3, rolename=CXO), Role(id=2, rolename=CEO), Role(id=1, rolename=CTO)]

==>  Preparing: SELECT rolename FROM role WHERE ( ( rolename like ? ) ) 
==> Parameters: %C%(String)
<==    Columns: rolename
<==      Total: 3
[Role(id=null, rolename=CTO), Role(id=null, rolename=CEO), Role(id=null, rolename=CXO)]

4.4 Mybatis Plus

MyBatis-Plus(简称 MP)是一个 MyBatis 的增强工具,在 MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。

官网地址:https://mp.baomidou.com

image-20200615125853362.png

5. Spring Boot 整合

5.1 Spring Boot 整合 Mybatis

Spring Boot 整个 Mybatis、PageHelper分页插件

项目整体目录结构

image-20200615113509111.png

5.1.1 添加依赖


    spring-boot-starter-parent
    org.springframework.boot
    2.1.14.RELEASE



    
        org.springframework.boot
        spring-boot-starter-web
    

    
        org.springframework.boot
        spring-boot-starter-test
    

    
        org.mybatis.spring.boot
        mybatis-spring-boot-starter
        2.1.1
    

    
        com.github.pagehelper
        pagehelper-spring-boot-starter
        1.2.12
    

    
        com.alibaba
        druid-spring-boot-starter
        1.1.21
    

    
        mysql
        mysql-connector-java
        5.1.49
    

    
        org.apache.commons
        commons-lang3
    

    
        org.projectlombok
        lombok
    


5.1.2 配置文件信息

spring:
  datasource:
    type: com.alibaba.druid.pool.DruidDataSource
    url: jdbc:mysql://luojie.site:33306/demo?serverTimezone=UTC&useUnicode=true&characterEncoding=utf-8&useSSL=false
    username: root
    password: 123456
    druid:
      #初始化时建立物理连接的个数
      initial-size: 10
      #最小连接池数量
      min-idle: 10
      #最大连接池数量
      max-active: 100
      #获取连接时最大等待时间
      max-wait: 60000
      stat-view-servlet:
        #配置监控页面访问登录名称
        login-username: admin
        #配置监控页面访问密码
        login-password: admin
      filter:
        stat:
          #是否开启慢sql查询监控
          log-slow-sql: true
          #慢SQL执行时间
          slow-sql-millis: 2

mybatis:
  # 配置XML配置文件中实例类别名路径
  type-aliases-package: org.example.mybatis.demo.domain
  configuration:
    # 开启驼峰命名规则
    map-underscore-to-camel-case: true
  # 配置MyBatis的xml配置文件路径
  mapper-locations: classpath:mapper/*Mapper.xml
logging:
  level:
    org.example: debug

5.1.3 映射文件信息

UserMapper.xml





    


5.1.4 相关代码

POJO 对象

@Data
public class User {
    private Integer id;
    private String username;
    private String password;
}

UserMapper

@Mapper
public interface UserMapper {

    List selectAll();

}

UserService

public interface UserService {

    List findUserAll();

    PageInfo page(int page, int size);
}

UserServiceImpl

@Service
public class UserServiceImpl implements UserService {

    @Autowired
    private UserMapper userMapper;

    @Override
    public List findUserAll() {
        return userMapper.selectAll();
    }

    @Override
    public PageInfo page(int page, int size) {
        PageHelper.startPage(page, size);
        List users = userMapper.selectAll();
        PageInfo pageInfo = new PageInfo<>(users);
        return pageInfo;
    }

}

5.1.5 测试代码

@RunWith(SpringRunner.class)
@SpringBootTest(classes = MybatisApplication.class)
public class MybatisTest {

    @Autowired
    private UserService userService;

    @Test
    public void test() {
        List userAll = userService.findUserAll();
        System.out.println(userAll);
    }

    @Test
    public void page() {
        PageInfo page = userService.page(1,2);
        System.out.println(page);
    }

}

你可能感兴趣的:(Mybatis笔记)