Mybatis源码系列3-三种SqlSession的区别(上)

简介: Mybatis源码系列3-三种SqlSession的区别(上)

自律,是一个人最好的修行

三个SqlSessionDefaultSqlSessionSqlSessionManager1.获取DefaultSqlSession的能力2.解决DefaultSqlSession的不足2.1解决自动关闭问题2.1解决线程安全问题。SqlSessionTemplate1.解决线程安全问题2.解决自动关闭问题总结


三个SqlSession


DefaultSqlSession与SqlSessionManager 与SqlSessionTemplate 是我常见的3种sqlsesion


image.png


从类图可以看出他们三个都实现了了SqlSession,也就是他们都可以表示一个会话。与其他不同的是SqlSessionManager实现了SqlSessionFactory

这三种sqlsession的区别是啥?他们的应用场景是啥呢?

这一切我们从DefaultSqlSession开始说起。


DefaultSqlSession


DefaultSqlSession 是SqlSession的默认实现。

当我们单独使用Mybatis时,我们通常使用DefaultSqlSession 来执行SQL,操作数据库。

但是DefaultSqlSession存在两个不足。

  1. 我们需要自己手动关闭sqlsesion,我们知道,人总是不可靠的。忘关sqlsession 是有很大概率发生的
  2. 线程安全问题:DefaultSqlSession是线程不安全的Sqlsession 。也就是说DefaultSqlSession不能是单例,

如何解决这两个问题?


自动关闭Session问题

  1. 我可以自己做一个切面,专门处理session关闭问题
  2. Mybatis为我们提供了升级版的DefaultSqlSession, SqlSessionManager可以解决这个问题


线程安全问题

  1. 既然不能共用,很自然的,我们每次使用DefaultSqlSession的时候都从SqlSessionFactory当中获取一个就可以了啊。
  2. 但是我们仍然想使用单例版的Sqlsession怎么办?别慌,Mybatis为我们提供了升级版的DefaultSqlSession ,SqlSessionManager可以解决这个问题。


SqlSessionManager


个人认为,与其说SqlSessionManager是DefaultSqlSession 的升级版,不如说SqlSessionManager是DefaultSqlSession代理版(或者封装版)

为什么这样说?来看看SqlSessionManager能干吗


1.获取DefaultSqlSession的能力

String resource = "mybatis-config.xml";
    // 读取配置文件
    InputStream inputStream = Resources.getResourceAsStream(resource);
    SqlSessionManager sqlSessionManager =SqlSessionManager.newInstance(inputStream);
    //获取一个SqlSession
    SqlSession session = sqlSessionManager.openSession();
    //SqlSessionManager 类的openSession
      public SqlSession openSession() {
        return sqlSessionFactory.openSession();
      }


从这个角度看,其实他跟SqlSession 没有什么区别。他只是封装了SqlSession ,具体工作还是交给SqlSession去做的.

sqlSessionManager.openSession() =  sqlSessionFactory.openSession()

得到的Sqlsession自然也是DefaultSqlSession


2.解决DefaultSqlSession的不足


2.1解决自动关闭问题

为了解决自动关闭问题,SqlSessionManager使用了代理技术来实现了自动关闭问题。


使用JDK动态代理技术,动态生成代理对象sqlSessionProxy ,并用内部类SqlSessionInterceptor来对SqlSession的执行方法进行增强。


private SqlSessionManager(SqlSessionFactory sqlSessionFactory) {
        this.sqlSessionFactory = sqlSessionFactory;
        //使用JDK代理技术,生成一个代理对象
        this.sqlSessionProxy = (SqlSession) Proxy.newProxyInstance(
            SqlSessionFactory.class.getClassLoader(),
            new Class[]{SqlSession.class},
            new SqlSessionInterceptor());
      }


我们以insert为例来看看


sqlSessionManager.insert()
    public int insert(String statement) {
        return sqlSessionProxy.insert(statement);
    }



当执行insert()方法时,sqlSessionManager内部是调用sqlsessionProxy代理对象的insert方法。之后执行增强器的SqlSessionInterceptor#invoke方法。



@Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
          final SqlSession sqlSession = SqlSessionManager.this.localSqlSession.get();
          if (sqlSession != null) {// 当使用线程本地变量
            try {
              return method.invoke(sqlSession, args);
            } catch (Throwable t) {
              throw ExceptionUtil.unwrapThrowable(t);
            }
          } else {//不使用线程本地变量。
              //从sqlSessionFactory获取一个DefaultSqlSession
            final SqlSession autoSqlSession = openSession();
            try {
              final Object result = method.invoke(autoSqlSession, args);
              autoSqlSession.commit();//提交
              return result;
            } catch (Throwable t) {
              autoSqlSession.rollback();//回滚
              throw ExceptionUtil.unwrapThrowable(t);
            } finally {
              autoSqlSession.close();//关闭sqlsession
            }
          }
        }
      }

invoke方法内部,调用openSession() 从sqlSessionFactory中获取一个DefaultSqlSession,执行对应的方法,并在finally中执行关闭sqlsession

最后的执行时序


image.png


sqlSessionManager.insert() 的背后依然是DefaultSqlSession.insert 。并且帮助我们close 了DefaultSqlSession。


开发人员再也不必担心,忘记关闭DefaultSqlSession 了。

在执行sqlSessionManager的sqlsession方法时, 其本质也是每次都创建DefaultSqlSession ,不正是线程安全的吗?


单例是sqlSessionManager ;但真正执行的是DefaultSqlSession



相关文章
|
4月前
|
SQL XML Java
mybatis-源码深入分析(一)
mybatis-源码深入分析(一)
|
2月前
|
SQL 安全 Java
MyBatis(6)#{}和${}的区别
在MyBatis中,`#{}`和`${}`是用于在SQL语句中嵌入参数的两种方式。`#{}`用于预处理参数,可以防止SQL注入;而`${}`进行直接字符串替换,适用于动态插入表名或列名,但存在SQL注入风险。建议优先使用`#{}`,并在必要时谨慎使用`${}`。
|
4月前
|
SQL XML Java
mybatis复习02,简单的增删改查,@Param注解多个参数,resultType与resultMap的区别,#{}预编译参数
文章介绍了MyBatis的简单增删改查操作,包括创建数据表、实体类、配置文件、Mapper接口及其XML文件,并解释了`#{}`预编译参数和`@Param`注解的使用。同时,还涵盖了resultType与resultMap的区别,并提供了完整的代码实例和测试用例。
mybatis复习02,简单的增删改查,@Param注解多个参数,resultType与resultMap的区别,#{}预编译参数
|
3月前
|
前端开发 Java 数据库连接
表白墙/留言墙 —— 中级SpringBoot项目,MyBatis技术栈MySQL数据库开发,练手项目前后端开发(带完整源码) 全方位全步骤手把手教学
本文是一份全面的表白墙/留言墙项目教程,使用SpringBoot + MyBatis技术栈和MySQL数据库开发,涵盖了项目前后端开发、数据库配置、代码实现和运行的详细步骤。
97 0
表白墙/留言墙 —— 中级SpringBoot项目,MyBatis技术栈MySQL数据库开发,练手项目前后端开发(带完整源码) 全方位全步骤手把手教学
|
3月前
|
Java 数据库连接 mybatis
Springboot整合Mybatis,MybatisPlus源码分析,自动装配实现包扫描源码
该文档详细介绍了如何在Springboot Web项目中整合Mybatis,包括添加依赖、使用`@MapperScan`注解配置包扫描路径等步骤。若未使用`@MapperScan`,系统会自动扫描加了`@Mapper`注解的接口;若使用了`@MapperScan`,则按指定路径扫描。文档还深入分析了相关源码,解释了不同情况下的扫描逻辑与优先级,帮助理解Mybatis在Springboot项目中的自动配置机制。
228 0
Springboot整合Mybatis,MybatisPlus源码分析,自动装配实现包扫描源码
|
5月前
|
XML Java 数据库连接
mybatis源码研究、搭建mybatis源码运行的环境
这篇文章详细介绍了如何搭建MyBatis源码运行的环境,包括创建Maven项目、导入源码、添加代码、Debug运行研究源码,并提供了解决常见问题的方法和链接到搭建好的环境。
mybatis源码研究、搭建mybatis源码运行的环境
|
5月前
|
Web App开发 前端开发 关系型数据库
基于SpringBoot+Vue+Redis+Mybatis的商城购物系统 【系统实现+系统源码+答辩PPT】
这篇文章介绍了一个基于SpringBoot+Vue+Redis+Mybatis技术栈开发的商城购物系统,包括系统功能、页面展示、前后端项目结构和核心代码,以及如何获取系统源码和答辩PPT的方法。
|
5月前
|
供应链 前端开发 Java
服装库存管理系统 Mybatis+Layui+MVC+JSP【完整功能介绍+实现详情+源码】
该博客文章介绍了一个使用Mybatis、Layui、MVC和JSP技术栈开发的服装库存管理系统,包括注册登录、权限管理、用户和货号管理、库存管理等功能,并提供了源码下载链接。
服装库存管理系统 Mybatis+Layui+MVC+JSP【完整功能介绍+实现详情+源码】
|
5月前
|
缓存 Java 数据库连接
我要手撕mybatis源码
该文章深入分析了MyBatis框架的初始化和数据读写阶段的源码,详细阐述了MyBatis如何通过配置文件解析、建立数据库连接、映射接口绑定、动态代理、查询缓存和结果集处理等步骤实现ORM功能,以及与传统JDBC编程相比的优势。
我要手撕mybatis源码
|
5月前
|
SQL Java 数据库连接
MyBatis 和 Hibernate 有什么区别?
【8月更文挑战第21天】
87 0