详解AbstractRoutingDataSource(动态数据源切换)实现原理(转载)

转载地址:详解AbstractRoutingDataSource(动态数据源切换)实现原理


多数据源让人最头痛的,不是配置多个数据源,而是如何能灵活动态的切换数据源。例如在一个springhibernate的框架的项目中,我们在spring配置中往往是配置一个dataSource来连接数据库,然后绑定给sessionFactory,在dao层代码中再指定sessionFactory来进行数据库操作。
在这里插入图片描述


上图是单数据源dataSource结构。但是上图的缺点太明显了,不支持多个数据源,于是我们再改进一下,让它支持多数据源。

这种结构实现了多数据源,但是缺点也很明显啊,具有多个SessionFactory不具有灵活性,而且太笨重了。如果再加一个数据源,就需要再加一个SessionFactory


SessionFactory接口负责初始化Hibernate。它充当数据存储源的代理,并负责创建Session对象。这里用到了工厂模式。需要注意的是SessionFactory并不是轻量级的。

顾名思义,SessionFactory,就是用来创建session会话(具体接下来讲)的工厂。如果存在多个Sessionfactory 那么Session是不是就乱套了,因此这种架构不可取。那么下面这种架构就应用而生。

详解AbstractRoutingDataSource(动态数据源切换)实现原理

看看上图,是不是感觉合理了很多。SpringAbstractRoutingDataSource就是采用这种架构。


看到这里你应该就明白了 AbstractRoutingDataSource 的原理。如果你还不明白,没关系,我们接着往下看。

看看?AbstractRoutingDataSource 的设计源码吧!

public abstract class AbstractRoutingDataSource extends AbstractDataSource implements InitializingBean {
……
}

扩展Spring的AbstractRoutingDataSource抽象类(该类充当了DataSource的路由中介, 能有在运行时, 根据某种key值来动态切换到真正的DataSource上。)

从上可以看出它继承了AbstractDataSource,而AbstractDataSource不就是javax.sql.DataSource的子类吗,So我们可以分析下它的getConnection方法:

 public Connection getConnection() throws SQLException {       
 	return determineTargetDataSource().getConnection();   
 }   
 public Connection getConnection(String username, String password)  throws SQLException {        
	 return determineTargetDataSource().getConnection(username, password);   
 }

获取连接的方法中,重点是determineTargetDataSource方法,看源码:

protected DataSource determineTargetDataSource() {
	Assert.notNull(this.resolvedDataSources, "DataSource router not initialized");
	Object lookupKey = determineCurrentLookupKey();
	DataSource dataSource = this.resolvedDataSources.get(lookupKey);
	if (dataSource == null && (this.lenientFallback || lookupKey == null)) {
		dataSource = this.resolvedDefaultDataSource;
	}
	if (dataSource == null) {
		throw new IllegalStateException("Cannot determine target DataSource for lookup key [" + lookupKey + "]");
	}
	return dataSource;
}

上面这段源码的重点在于determineCurrentLookupKey()方法,这是AbstractRoutingDataSource类中的一个抽象方法,而它的返回值是你所要用的数据源dataSourcekey值,有了这个key值,resolvedDataSource(这是个map,由配置文件中设置好后存入的)就从中取出对应的DataSource,如果找不到,就用配置默认的数据源。

看完源码,应该有点启发了吧,没错!你要扩展AbstractRoutingDataSource类,并重写其中的determineCurrentLookupKey方法,来实现数据源的切换:

 import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource; 
 /**  
  * 获取数据源(依赖于spring)  
  * @author linhy  
  */ 
 public class DynamicDataSource extends AbstractRoutingDataSource{     
 
 @Override
 protected Object determineCurrentLookupKey() {         
 	return DataSourceHolder.getDataSource();     
 	} 
 }
 

参考

猜你喜欢

转载自blog.csdn.net/qq_31156277/article/details/85227415