动态代理的楷模:源码分析Mybatis与Spring(二)

前言


上一篇介绍了Mapper接口代理的实现原理,今天继续剖析SqlSession代理。希望看完大家能搞懂下面问题:

  • Spring是如何管理Mapper的Bean,实现线程安全
  • Mybatis自身的sqlSession是否线程安全


    spring-mybatis.png

源码分析


二. SqlSession代理

  • 1、创建SqlSessionTemplate

在Spring扫描Mapper创建bean时,可以留意到不仅设置了className,beanClass,还设置了sqlSessionFactory属性。

public class ClassPathMapperScanner extends ClassPathBeanDefinitionScanner {

    private MapperFactoryBean<?> mapperFactoryBean = new MapperFactoryBean<Object>();
    
    private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitions) {
        GenericBeanDefinition definition;
        for (BeanDefinitionHolder holder : beanDefinitions) {
            definition = (GenericBeanDefinition) holder.getBeanDefinition();
            //将mapper接口的名称添加到构造参数
         definition.getConstructorArgumentValues().addGenericArgumentValue(definition.getBeanClassName());
            //设置BeanDefinition的class
            definition.setBeanClass(this.mapperFactoryBean.getClass());
            //添加属性addToConfig
            definition.getPropertyValues().add("addToConfig", this.addToConfig);
            //添加属性sqlSessionFactory
            definition.getPropertyValues().add("sqlSessionFactory", new RuntimeBeanReference(this.sqlSessionFactoryBeanName));
            ......
    }
}

MapperFactoryBean继承于SqlSessionDaoSupport,当设置sqlSessionFactory属性时会触发setSqlSessionFactory()方法。

public abstract class SqlSessionDaoSupport extends DaoSupport {

  private SqlSessionTemplate sqlSessionTemplate;

  /**
   * Set MyBatis SqlSessionFactory to be used by this DAO. Will automatically create SqlSessionTemplate for the given
   * SqlSessionFactory.
   *
   * @param sqlSessionFactory
   *          a factory of SqlSession
   */
  public void setSqlSessionFactory(SqlSessionFactory sqlSessionFactory) {
    if (this.sqlSessionTemplate == null || sqlSessionFactory != this.sqlSessionTemplate.getSqlSessionFactory()) {
      this.sqlSessionTemplate = createSqlSessionTemplate(sqlSessionFactory);
    }
  protected SqlSessionTemplate createSqlSessionTemplate(SqlSessionFactory sqlSessionFactory) {
    return new SqlSessionTemplate(sqlSessionFactory);
  } 
}

最终会为每个Mapper Bean创建一个SqlSessionTemplate对象,因此SqlSessionTemplate在Mapper Bean之间是不共享的。

  • 2、创建动态代理
    再看看SqlSessionTemplate的构造函数
public class SqlSessionTemplate implements SqlSession, DisposableBean {

  public SqlSessionTemplate(SqlSessionFactory sqlSessionFactory, ExecutorType executorType,PersistenceExceptionTranslator exceptionTranslator) {
       ...
      this.sqlSessionFactory = sqlSessionFactory;
      this.executorType = executorType;
      this.exceptionTranslator = exceptionTranslator;
      //创建sqlSession代理
      this.sqlSessionProxy = (SqlSession) newProxyInstance(SqlSessionFactory.class.getClassLoader(),
        new Class[] { SqlSession.class }, new SqlSessionInterceptor());
  }

SqlSessionTemplate是实现了SqlSession接口的类。回顾上一篇,Mapper Bean实际调用的代理类MapperProxy,invoke()方法里用到的sqlSession就是SqlSessionTemplate。然而SqlSessionTemplate自己也不亲自操作,而是使用动态代理替自己完成sqlSession操作

public class SqlSessionTemplate implements SqlSession, DisposableBean {
  
  private final SqlSession sqlSessionProxy;
  ...
  @Override
  public void select(String statement, Object parameter, RowBounds rowBounds, ResultHandler handler) {
    this.sqlSessionProxy.select(statement, parameter, rowBounds, handler);
  }
  @Override
  public int insert(String statement) {
    return this.sqlSessionProxy.insert(statement);
  }
  ...
}

顺而言之,关注点来到处理类SqlSessionInterceptor。它是SqlSessionTemplate私有内部类。

 private class SqlSessionInterceptor implements InvocationHandler {
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
      //获取当前线程事务的SqlSession
      SqlSession sqlSession = getSqlSession(SqlSessionTemplate.this.sqlSessionFactory,
          SqlSessionTemplate.this.executorType, SqlSessionTemplate.this.exceptionTranslator);
      try {
        Object result = method.invoke(sqlSession, args);
        if (!isSqlSessionTransactional(sqlSession, SqlSessionTemplate.this.sqlSessionFactory)) {
          sqlSession.commit(true);
        }
        return result;
      } catch (Throwable t) {
        ...
      } finally {
        if (sqlSession != null) {
          closeSqlSession(sqlSession, SqlSessionTemplate.this.sqlSessionFactory);
        }
      }
    }
  }
  • 3、sqlSession线程管理
    SqlSessionInterceptor的invoke方法获取Sqlsession的方式并不简单。奥妙在于getSqlSession()方法。它是Spring专门管理sqlSession的工具类SqlSessionUtils里的静态方法。
public final class SqlSessionUtils {

    public static SqlSession getSqlSession(SqlSessionFactory sessionFactory, ExecutorType executorType,PersistenceExceptionTranslator exceptionTranslator) {
        ...
        //从当前线程获取SqlSessionHolder
        SqlSessionHolder holder = (SqlSessionHolder) TransactionSynchronizationManager.getResource(sessionFactory);
        //从SqlSessionHolder获取SqlSession,引用次数加1
        SqlSession session = sessionHolder(executorType, holder);
        if (session != null) {
          return session;
        }

        LOGGER.debug(() -> "Creating a new SqlSession");
        //如果当前线程没有sqlSession,创建一个
        session = sessionFactory.openSession(executorType);
       //将新的sqlSession注册当前线程
        registerSessionHolder(sessionFactory, executorType, exceptionTranslator, session);

        return session;
    }
}

可以看出sqlSession是由SqlSessionHolder管理的。只要搞懂sqlSession如何创建与注册,获取的事就再简单不过了。重心放在最后两行代码sessionFactory.openSession(executorType)registerSessionHolder(sessionFactory, executorType, exceptionTranslator, session);

  • 4、openSession()
    实现了SqlSessionFactory的接口有两个:DefaultSqlSessionFactory和SqlSessionManager。
    SqlSessionManager还实现了SqlSession类,是供用户单独用Mybatis时使用的。MyBatis为了将SqlSession管理权交给Spring。将SqlSession由SqlSessionTemplate来实现,SqlSessionFactory使用DefaultSqlSessionFactory。


    SqlSession.png
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);
          //新建一个DefaultSqlSession对象作为SqlSession
          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();
        }
      }
}

DefaultSqlSessionFactory不仅创建sqlSession还创建了事务。

  • 5、registerSessionHolder
private static void registerSessionHolder(SqlSessionFactory sessionFactory, ExecutorType executorType,PersistenceExceptionTranslator exceptionTranslator, SqlSession session) {
      SqlSessionHolder holder;
        ...  
        Environment environment = sessionFactory.getConfiguration().getEnvironment();
        ...
        //将新的sqlSession与sqlSessionHolder绑定
        holder = new SqlSessionHolder(session, executorType, exceptionTranslator);
        //将新的sqlSessionHolder与线程绑定
        TransactionSynchronizationManager.bindResource(sessionFactory, holder);
        TransactionSynchronizationManager
            .registerSynchronization(new SqlSessionSynchronization(holder, sessionFactory));
        holder.setSynchronizedWithTransaction(true);
        //引用次数加一
        holder.requested();
        ...
      }

registerSessionHolder是SqlSessionUtils的私有静态方法,会创建新的SqlSessionHolder与当前线程绑定。

public abstract class TransactionSynchronizationManager {

    private static final ThreadLocal<Map<Object, Object>> resources = new NamedThreadLocal<>("Transactional resources");

    public static void bindResource(Object key, Object value) throws IllegalStateException {
        Object actualKey = TransactionSynchronizationUtils.unwrapResourceIfNecessary(key);
        Assert.notNull(value, "Value must not be null");
        Map<Object, Object> map = resources.get();
        // set ThreadLocal Map if none found
        if (map == null) {
            map = new HashMap<>();
            resources.set(map);
        }
        Object oldValue = map.put(actualKey, value);
    }
}

bindResource()实现原理也很简单。将SessionFactory作为key,SqlSessionHolder为value存入一个map里,然后再将这个map存入ThreadLocal。因此,Spring调用Mybatis每个线程调用的是不同的SqlSessionHolder即不同的SqlSession,Spring就是这样管理sqlSession,实现线程安全的。

总结


经过两篇文章对mybatis源码的层层剖析,可知。Mapper Bean是通过MapperFactoryBean创建MapperProxy代理获取实现类。MapperProxy用到的SqlSession是SqlSessionTemplate,实际执行的是代理处理类SqlSessionInterceptor。执行时会从当前线程获取SqlSessionHolder中的SqlSession,实现线程安全。

MapperBean.png

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,110评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,443评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,474评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,881评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,902评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,698评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,418评论 3 419
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,332评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,796评论 1 316
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,968评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,110评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,792评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,455评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,003评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,130评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,348评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,047评论 2 355

推荐阅读更多精彩内容