需要的背景知识:Spring 和Mybatis 实现原理和源码, javaassist字节码增强的使用, java及设计模式的使用

1 读取解析数据库配置文件

DataSourceScanner实现了Spring的BeanDefinitionRegistryPostProcessor和ApplicationContextAware接口。

BeanDefinitionRegistryPostProcessor 允许定义bean

ApplicationContextAware 可以获取Spring上下文。

代码流程:getDataSources解析读取数据库配置文件生成dataSourcesMapping( Map<String, Map<DataSourceType, DataSource>>),registerDataSources 遍历dataSourcesMapping生成DataSourceFactoryBean(名字格式trade0005SlaveDataSource)并注入到SpringBean工厂。继续遍历解析数据源,封装成Map<String, ReadWriteSplittingDataSource> ,如果有可写数据源则创建事务管理器(名字格式trade0005TransactionManager),注册到SpringBean工厂。如果可写数据源不存在,注册一个空transactionManager。如果只有一个可写数据源,则添加别名,兼容默认情况。解析完数据源后,向Spring注册dataSourceLookup,其实就是把数据源map保存起来。

2 MapperScannerWithSharding

  Tsharding MybatisMapper的扫描类,负责将Mapper接口与对应的xml配置文件整合,绑定设定的数据源,注入到Spring Context中。

  实现了BeanFactoryPostProcessor, InitializingBean 接口  。BeanFactoryPostProcessor可以修改BEAN的配置信息

  代码流程:

  获取1步中注册的dataSourceLookup。

    initSqlSessionFactories。第一次SqlSessionFactoryBean.setMapperLocations.setDataSource.setTypeAliasesPackage(mapper位置,数据源,实体类),实例化SqlSessionFactoryBean,单例注册到bean工厂(名字格式为trade0001)。后续,从第一次的sessionFactoryBean.getObject()获取一个SqlSessionFactory,

然后从SqlSessionFactory获取configuration,就是Resource,

从Resource获取SqlSessionFactory返回Configuration,从Configuration获取mappedStatements( Map<String, MappedStatement>),

然后为了后续的元数据复用再 Map<String, MappedStatement> 放回Configuration.最后使用SqlSessionFactoryLookup 把所有sqlSessionFactories保存起来,放入MapperScannerWithSharding

  ClassPathScanHandler:扫描所有配置文件下所以java类,

新生成mapper,如果这些类是mapper类,新生成,并单例注册到bean工厂

  

    private Object newMapper(final Class<?> clazz) {

        final Invoker invoker = new TShardingRoutingInvokeFactory(sqlSessionFactoryLookup).newInvoker(clazz);
  //动态代理 invoker
return Proxy.newProxyInstance(clazz.getClassLoader(), new Class[]{clazz},
new InvocationHandler() {
@Override
public Object invoke(Object proxy, final Method method, final Object[] args) throws Throwable {
return invoker.invoke(new DefaultInvocation(method, args));
}
});
}

  

TShardingRoutingInvokeFactory 类:
 判断mapperInterface 是否带注解,如果注解为TShardingRoutingHandler,使用Sharding数据源,走分表分库。
不带注解,正常的使用配置的数据源。 走分表分库分支:返回内部类Invoker

3:MapperShardingInitializer 增强Mapper处理总入口:Mapper被mybatis初始化后,在这里做进一步的处理和增强、

  http://www.cnblogs.com/clds/p/5966815.html

    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
Map<String, SqlSessionFactory> sqlSessionFactories = applicationContext.getBeansOfType(SqlSessionFactory.class);
if (sqlSessionFactories.isEmpty()) {
return;
}
MapperHelperForSharding mapperHelperForSharding = new MapperHelperForSharding();
List<SqlSession> sqlSessions = new ArrayList<>(sqlSessionFactories.size());
for (SqlSessionFactory sqlSessionFactory : sqlSessionFactories.values()) {
SqlSession sqlSession = new SqlSessionTemplate(sqlSessionFactory);
sqlSessions.add(sqlSession);
}
//Mapper代码增强 每个方法扩展出一个ShardingMapper类,增强为512个方法。
this.needEnhancedClassesArray = needEnhancedClasses.split(",");
this.enhanceMapperClass();
mapperHelperForSharding.setMappers(needEnhancedClassesArray);
mapperHelperForSharding.setSqlSessions(sqlSessions.toArray(new SqlSession[0]));
mapperHelperForSharding.initMapper();
}

4:执行dao方法
  因为是动态代理,又返回到TShardingRoutingInvokeFactory。

  使用动态代理的原因:Object mapper = this.newMapper(clazz);

    beanFactory.registerSingleton(Character.toLowerCase(clazz.getSimpleName().charAt(0))+ clazz.getSimpleName().substring(1), mapper);
  bean工厂中只注入了shopOrderMapper ,但是实际执行的时候需要找到增强后的方法

new Invoker() {
@Override
public Object invoke(Invocation invocation) throws Throwable { Method method = invocation.getMethod();
ShardingMetadata shardingMetadata = getShardingKey(method, invocation.getArgs());//根据分片参数获取schemaName+tableSuffix if (shardingMetadata == null) {
throw new DataSourceRoutingException("dataSourceRouting error! Method Name:" + method.getName() + " shardingMetadata is null!");
} //走分库分表环境
logger.debug("TShardingRoutingInvokeFactory routing to sharding db. Method Name:" + method.getName() + ". ShardingKey:" + shardingMetadata.getShardingKey()); Class newClass = clazz;
if (!"".equals(shardingMetadata.getSchemaName())) {
newClass = Class.forName(clazz.getCanonicalName() + "Sharding" + method.getName());//找到对应增强后的class
}
            ////newMethod:getShopOrderByShopOrderIds0064
Method newMethod = newClass.getMethod(method.getName() + shardingMetadata.getTableSuffix(), method.getParameterTypes());
MapperBasicConfig config = new MapperBasicConfig(newClass, shardingMetadata.getSchemaName());//应该走哪个数据源
final Object mapper = newMyBatisMapper(config);//获取到对应的mapper ShopOrderMapperShardinggetShopOrderByShopOrderIds
try {
ReadWriteSplittingContextInitializer.initReadWriteSplittingContext(invocation.getMethod());//框架会回调此方法,判断数据源类型,获取数据源类型 master or slave
return newMethod.invoke(mapper, invocation.getArgs());//执行dao方法
} finally {
ReadWriteSplittingContextInitializer.clearReadWriteSplittingContext();
}
}
}

TSharding源码阅读的更多相关文章

  1. TSharding源码阅读-MapperShardingInitializer

    /** * 增强Mapper处理总入口:Mapper被mybatis初始化后,在这里做进一步的处理和增强 * * @author qigong on 5/1/15 */ public class Ma ...

  2. 【原】FMDB源码阅读(三)

    [原]FMDB源码阅读(三) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 FMDB比较优秀的地方就在于对多线程的处理.所以这一篇主要是研究FMDB的多线程处理的实现.而 ...

  3. 【原】FMDB源码阅读(二)

    [原]FMDB源码阅读(二) 本文转载请注明出处 -- polobymulberry-博客园 1. 前言 上一篇只是简单地过了一下FMDB一个简单例子的基本流程,并没有涉及到FMDB的所有方方面面,比 ...

  4. 【原】FMDB源码阅读(一)

    [原]FMDB源码阅读(一) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 说实话,之前的SDWebImage和AFNetworking这两个组件我还是使用过的,但是对于 ...

  5. 【原】AFNetworking源码阅读(六)

    [原]AFNetworking源码阅读(六) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 这一篇的想讲的,一个就是分析一下AFSecurityPolicy文件,看看AF ...

  6. 【原】AFNetworking源码阅读(五)

    [原]AFNetworking源码阅读(五) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇中提及到了Multipart Request的构建方法- [AFHTTP ...

  7. 【原】AFNetworking源码阅读(四)

    [原]AFNetworking源码阅读(四) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇还遗留了很多问题,包括AFURLSessionManagerTaskDe ...

  8. 【原】AFNetworking源码阅读(三)

    [原]AFNetworking源码阅读(三) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇的话,主要是讲了如何通过构建一个request来生成一个data tas ...

  9. 【原】AFNetworking源码阅读(二)

    [原]AFNetworking源码阅读(二) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇中我们在iOS Example代码中提到了AFHTTPSessionMa ...

随机推荐

  1. MySQL Cluster导入数据表时报错:Got error 708 'No more attribute metadata records (increase MaxNoOfAttributes)' from NDBCLUSTER

    准备把以前的非集群版MySQL数据导入到MySQL Cluster中,出现 'No more attribute metadata records (increase MaxNoOfAttribute ...

  2. 天地图应用ArcGIS发布的服务(转)

    天地图应用ArcGIS发布的服务 本文包含三个部分:利用ArcMap将Excel的数据转化为ArcGIS MXD文件.利用ArcMap发布服务.天地图添加ArcGIS发布的服务. 一 MXD文件的生成 ...

  3. log4j教程 5、示例程序

    前面我们已经看到了如何创建一个配置文件.本教程将讲解如何生成调试信息和日志在一个简单的文本文件. 下面是我们的例子中创建了一个简单的配置文件.这里再重复一次: 下载最新的Log4j库:http://l ...

  4. redhat mount iso as one yum repository

    prepare redhat DVD iso rhel-server-6.4-x86_64-dvd.iso mount cd / mkdir /mnt/rhel mount -o loop rhel- ...

  5. 对中级 Linux 用户非常有用的 20 个命令

    FROM:http://www.oschina.net/translate/20-advanced-commands-for-middle-level-linux-users 21. 命令: Find ...

  6. setlocal启动批处理文件中环境变量的本地化

    setlocal启动批处理文件中环境变量的本地化 在执行 SETLOCAL 之后所做的环境改动只限于批处理文件.要还原原先的设置,必须执行 ENDLOCAL. 学习了:https://baike.ba ...

  7. 7. JPA - Hibernate【从零开始学Spring Boot】

    转载:http://blog.csdn.net/linxingliang/article/details/51636976 在说具体如何在spring boot 使用hibernate前,先抛装引玉些 ...

  8. 不厚道一回->Omnifocus 2 for mac license

    rt, 发个Omnifocus 2 for mac license. 其实Omnifocus 2的价格已经还算亲民了..可惜手贱一下子就找到了,所以没买了..不敢独享,所以分享给需要的人..有能力还是 ...

  9. 如何安装Android SDK Emulator

    1 下载并安装JDK,可以到官方网站寻找自己的对应版本下载 http://www.oracle.com/technetwork/java/javase/downloads/jdk-7u1-downlo ...

  10. Wamp访问缓慢、本地主机访问缓慢解决方案

    Wamp访问缓慢.本地主机访问缓慢解决方案 Wamp访问速度缓慢可能的原因:          1.一般原因: Wamp的日志文件太大.处理办法,将位于wamp/logs/下的日志文件清空.     ...