博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Spring + iBatis 的多库横向切分简易解决思路
阅读量:6835 次
发布时间:2019-06-26

本文共 13004 字,大约阅读时间需要 43 分钟。

国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:
内部邀请码:
C8E245J (不写邀请码,没有现金送)
国内私募机构九鼎控股打造,九鼎投资是在全国股份转让系统挂牌的公众公司,股票代码为430719,为“中国PE第一股”,市值超1000亿元。 

 

------------------------------------------------------------------------------------------------------------------------------------------------------------------

 

 

原文地址:

1.引言 

   笔者最近在做一个互联网的“类SNS”应用,应用中用户数量巨大(约4000万)左右,因此,简单的使用传统单一数据库存储肯定是不行的。 
   参考了业内广泛使用的分库分表,以及使用DAL数据访问层等的做法,笔者决定使用一种最简单的数据源路由选择方式来解决问题。 
   严格的说,目前的实现不能算是一个解决方案,只能是一种思路的简易实现,笔者也仅花了2天时间来完成(其中1.5天是在看资料和Spring/ibatis的源码)。这里也只是为各位看官提供一个思路参考,顺便给自己留个笔记 
2.系统的设计前提 
   我们的系统使用了16个数据库实例(目前分布在2台物理机器上,后期将根据系统负荷的增加,逐步移库到16台物理机器上)。16个库是根据用户的UserID进行简单的hash分配。这里值得一说的是,我们既然做了这样的横向切分设计,就已经考虑了系统需求的特性, 

  • 1.不会发生经常性的跨库访问。
  • 2.主要的业务逻辑都是围绕UserID为核心的,在一个单库事务内即可完成。

   在系统中,我们使用Spring和iBatis。Spring负责数据库的事务管理AOP,以及Bean间的IOC。选择iBatis的最大原因是对Sql的性能优化,以及后期如果有分表要求的时,可以很容易实现对sql表名替换。 

3.设计思路 
   首先,要说明一下笔者的思路,其实很简单,即“在每次数据库操作前,确定当前要选择的数据库对象”而后就如同访问单库一样的访问当前选中的数据库即可。 
   其次,要在每次DB访问前选择数据库,需要明确几个问题,1.iBatis在什么时候从DataSource中取得具体的数据库Connection的,2.对取得的Connection,iBatis是否进行缓存,因为在多库情况下Connection被缓存就意味着无法及时改变数据库链接选择。3.由于我们使用了Spring来管理DB事务,因此必须搞清Spring对DB Connction的开关拦截过程是否会影响多DataSource的情况。 
   幸运的是,研究源码的结果发现,iBatis和Spring都是通过标准的DataSource接口来控制 
Connection的,这就为我们省去了很多的麻烦,只需要实现一个能够支持多个数据库的DataSource,就能达到我们的目标。 
4.代码与实现 
多数据库的DataSource实现:MultiDataSource.class 

 
import java.io.PrintWriter;import java.sql.Connection;import java.sql.SQLException;import java.util.ArrayList;import java.util.Collection;import java.util.HashMap;import java.util.Map;import javax.sql.DataSource;import org.apache.log4j.Logger;import com.xxx.sql.DataSourceRouter.RouterStrategy;/** * 复合多数据源(Alpha) * @author linliangyi2005@gmail.com * Jul 15, 2010 */public class MultiDataSource implements DataSource {		static Logger logger = Logger.getLogger(MultiDataSource.class);		//当前线程对应的实际DataSource	private ThreadLocal
currentDataSourceHolder = new ThreadLocal
(); //使用Key-Value映射的DataSource private Map
mappedDataSources; //使用横向切分的分布式DataSource private ArrayList
clusterDataSources; public MultiDataSource(){ mappedDataSources = new HashMap
(4); clusterDataSources = new ArrayList
(4); } /** * 数据库连接池初始化 * 该方法通常在web 应用启动时调用 */ public void initialMultiDataSource(){ for(DataSource ds : clusterDataSources){ if(ds != null){ Connection conn = null; try { conn = ds.getConnection(); } catch (SQLException e) { e.printStackTrace(); } finally{ if(conn != null){ try { conn.close(); } catch (SQLException e) { e.printStackTrace(); } conn = null; } } } } Collection
dsCollection = mappedDataSources.values(); for(DataSource ds : dsCollection){ if(ds != null){ Connection conn = null; try { conn = ds.getConnection(); } catch (SQLException e) { e.printStackTrace(); } finally{ if(conn != null){ try { conn.close(); } catch (SQLException e) { e.printStackTrace(); } conn = null; } } } } } /** * 获取当前线程绑定的DataSource * @return */ public DataSource getCurrentDataSource() { //如果路由策略存在,且更新过,则根据路由算法选择新的DataSource RouterStrategy strategy = DataSourceRouter.currentRouterStrategy.get(); if(strategy == null){ throw new IllegalArgumentException("DataSource RouterStrategy No found."); } if(strategy != null && strategy.isRefresh()){ if(RouterStrategy.SRATEGY_TYPE_MAP.equals(strategy.getType())){ this.choiceMappedDataSources(strategy.getKey()); }else if(RouterStrategy.SRATEGY_TYPE_CLUSTER.equals(strategy.getType())){ this.routeClusterDataSources(strategy.getRouteFactor()); } strategy.setRefresh(false); } return currentDataSourceHolder.get(); } public Map
getMappedDataSources() { return mappedDataSources; } public void setMappedDataSources(Map
mappedDataSources) { this.mappedDataSources = mappedDataSources; } public ArrayList
getClusterDataSources() { return clusterDataSources; } public void setClusterDataSources(ArrayList
clusterDataSources) { this.clusterDataSources = clusterDataSources; } /** * 使用Key选择当前的数据源 * @param key */ public void choiceMappedDataSources(String key){ DataSource ds = this.mappedDataSources.get(key); if(ds == null){ throw new IllegalStateException("No Mapped DataSources Exist!"); } this.currentDataSourceHolder.set(ds); } /** * 使用取模算法,在群集数据源中做路由选择 * @param routeFactor */ public void routeClusterDataSources(int routeFactor){ int size = this.clusterDataSources.size(); if(size == 0){ throw new IllegalStateException("No Cluster DataSources Exist!"); } int choosen = routeFactor % size; DataSource ds = this.clusterDataSources.get(choosen); if(ds == null){ throw new IllegalStateException("Choosen DataSources is null!"); } logger.debug("Choosen DataSource No." + choosen+ " : " + ds.toString()); this.currentDataSourceHolder.set(ds); } /* (non-Javadoc) * @see javax.sql.DataSource#getConnection() */ public Connection getConnection() throws SQLException { if(getCurrentDataSource() != null){ return getCurrentDataSource().getConnection(); } return null; } /* (non-Javadoc) * @see javax.sql.DataSource#getConnection(java.lang.String, java.lang.String) */ public Connection getConnection(String username, String password) throws SQLException { if(getCurrentDataSource() != null){ return getCurrentDataSource().getConnection(username , password); } return null; } /* (non-Javadoc) * @see javax.sql.CommonDataSource#getLogWriter() */ public PrintWriter getLogWriter() throws SQLException { if(getCurrentDataSource() != null){ return getCurrentDataSource().getLogWriter(); } return null; } /* (non-Javadoc) * @see javax.sql.CommonDataSource#getLoginTimeout() */ public int getLoginTimeout() throws SQLException { if(getCurrentDataSource() != null){ return getCurrentDataSource().getLoginTimeout(); } return 0; } /* (non-Javadoc) * @see javax.sql.CommonDataSource#setLogWriter(java.io.PrintWriter) */ public void setLogWriter(PrintWriter out) throws SQLException { if(getCurrentDataSource() != null){ getCurrentDataSource().setLogWriter(out); } } /* (non-Javadoc) * @see javax.sql.CommonDataSource#setLoginTimeout(int) */ public void setLoginTimeout(int seconds) throws SQLException { if(getCurrentDataSource() != null){ getCurrentDataSource().setLoginTimeout(seconds); } } /* (non-Javadoc) * 该接口方法since 1.6 * 不是所有的DataSource都实现有这个方法 * @see java.sql.Wrapper#isWrapperFor(java.lang.Class) */ public boolean isWrapperFor(Class
iface) throws SQLException { // if(getCurrentDataSource() != null){// return getCurrentDataSource().isWrapperFor(iface);// } return false; } /* (non-Javadoc) * 该接口方法since 1.6 * 不是所有的DataSource都实现有这个方法 * @see java.sql.Wrapper#unwrap(java.lang.Class) */ public
T unwrap(Class
iface) throws SQLException {// if(getCurrentDataSource() != null){// return getCurrentDataSource().unwrap(iface);// } return null; }

这个类实现了DataSource的标准接口,而最核心的部分是getConnection()方法的重载。下面具体阐述: 

  • 1.实例变量 clusterDataSources 是一个DataSource 的 ArrayList它存储了多个数据库的DataSource实例,我们使用Spring的IOC功能,将多个DataSource注入到这个list中。
  • 2.实例变量 mappedDataSources 是一个DataSource 的Map,它与clusterDataSources 一样用来存储多个数据库的DataSource实例,不同的是,它可以使用key直接获取DataSource。我们一样会使用Spring的IOC功能,将多个DataSource注入到这个Map中。
  • 3.实例变量currentDataSourceHolder ,他是一个ThreadLocal变量,保存与当前线程相关的且已经取得的DataSource实例。这是为了在同一线程中,多次访问同一数据库时,不需要再重新做路由选择。
  • 4.当外部类调用getConnection()方法时,方法将根据上下文的路由规则,从clusterDataSources 或者 mappedDataSources 选择对应DataSource,并返回其中的Connection。

(PS:关于DataSource的路由选择规则,可以根据应用场景的不同,自行设计。笔者这里提供两种简单的思路,1.根据HashCode,在上述例子中可以是UserId,进行取模运算,来定位数据库。2.根据上下文设置的关键字key,从map中选择映射的DataSource) 
5.将MultiDataSource与Spring,iBatis结合 
    在完成了上述的编码过程后,就是将这个MultiDataSource与现有Spring和iBatis结合起来配置。 
STEP 1。配置多个数据源 
笔者这里使用了C3P0作为数据库连接池,这一步和标准的Spring配置一样,唯一不同的是,以前只配置一个,现在要配置多个 

 
${jdbc.driverClass}
${mysql.url_1}
${jdbc.username}
${jdbc.password}
${c3p0.minPoolSize}
${c3p0.maxPoolSize}
${c3p0.initialPoolSize}
${c3p0.idleConnectionTestPeriod}
${jdbc.driverClass}
${mysql.url_2}
${jdbc.username}
${jdbc.password}
${c3p0.minPoolSize}
${c3p0.maxPoolSize}
${c3p0.initialPoolSize}
${c3p0.idleConnectionTestPeriod}
......

 

STEP 2。将多个数据源都注入到MultiDataSource中 

  

STEP 3。像使用标准的DataSource一样,使用MultiDataSource 

  

至此,我们的程序就可以让Spring来管理多库访问了,但请注意,数据库事务仍然限于单库范围(之前已经说过,这里的应用场景不存在跨库的事务)。 
6.Java代码使用例子 
首先要说明的是,这里我们只是提供了一个简单的使用范例,在范例中,我们还必须手动的调用API,以确定DataSource的路由规则,在实际的应用中,您可以针对自己的业务特点,对此进行封装,以实现相对透明的路由选择 

 
public boolean addUserGameInfo(UserGameInfo userGameInfo){		//1.根据UserGameInfo.uid 进行数据源路由选择		DataSourceRouter.setRouterStrategy(				RouterStrategy.SRATEGY_TYPE_CLUSTER ,				null,				userGameInfo.getUid());				//2.数据库存储		try {			userGameInfoDAO.insert(userGameInfo);			return true;		} catch (SQLException e) {			e.printStackTrace();			logger.debug("Insert UserGameInfo failed. " + userGameInfo.toString());		}		return false;	}

  

OK,我们的多库横向切分的实验可以暂告一个段落。实际上,要实现一个完整的DAL是非常庞大的工程,而对我们推动巨大的,可能只是很小的一个部分,到处都存在着8-2法则,要如何选择,就看各位看官了!! 

 

补充:

DataSourceRouter.java

 

/** * @author linliangyi2005@gmail.com * Jul 15, 2010 */public class DataSourceRouter {    public static ThreadLocal
currentRouterStrategy = new ThreadLocal
(); /** * 设置MultiDataSource的路由策略 * @param type * @param key * @param routeFactor */ public static void setRouterStrategy(String type , String key , int routeFactor){ if(type == null){ throw new IllegalArgumentException("RouterStrategy Type must not be null"); } RouterStrategy rs = currentRouterStrategy.get(); if(rs == null){ rs = new RouterStrategy(); currentRouterStrategy.set(rs); } rs.setType(type); rs.setKey(key); rs.setRouteFactor(routeFactor); } /** * 数据源路由策略 * @author linliangyi2005@gmail.com * Jul 15, 2010 */ public static class RouterStrategy{ public static final String SRATEGY_TYPE_MAP = "MAP"; public static final String SRATEGY_TYPE_CLUSTER = "CLUSTER"; /* * 可选值 “MAP” , “CLUSTER” * MAP : 根据key从DataSourceMap中选中DS * CLUSTER : 根据routeFactor参数,通过算法获取群集 */ private String type; /* * “MAP” ROUTE 中的key * */ private String key; /* * "CLUSTER" ROUTE时的参数 */ private int routeFactor; /* * True表示RouterStrategy更新过 * False表示没有更新 */ private boolean refresh; public String getType() { return type; } public void setType(String type) { if(this.type != null && !this.type.equals(type)){ this.type = type; this.refresh = true; }else if(this.type == null && type != null){ this.type = type; this.refresh = true; } } public String getKey() { return key; } public void setKey(String key) { if(this.key != null && !this.key.equals(key)){ this.key = key; this.refresh = true; }else if(this.key == null && key != null){ this.key = key; this.refresh = true; } } public int getRouteFactor() { return routeFactor; } public void setRouteFactor(int routeFactor) { if(this.routeFactor != routeFactor){ this.routeFactor = routeFactor; this.refresh = true; } } public boolean isRefresh() { return refresh; } public void setRefresh(boolean refresh) { this.refresh = refresh; } }}

 

 

 

转载地址:http://qrmkl.baihongyu.com/

你可能感兴趣的文章
对路径的访问被拒绝
查看>>
PHP 短连接生成
查看>>
lftp 4.4.0 发布,命令行的FTP工具
查看>>
uva 10152 ShellSort
查看>>
经典网页设计:20个与众不同的 Flash 网站设计作品
查看>>
java.util.concurrent.locks.Condition 例子程序探讨
查看>>
NYOJ 506
查看>>
解决读写Excel的第三方类库as3xls无法读取中文和写入中文的问题
查看>>
直线职权::参谋职权::职能职权
查看>>
转发:Hekaton:SQL Server集成的内存事务处理
查看>>
[转]孙鑫VC教程例子代码1---Windows程序内部运行原理
查看>>
手把手玩转win8开发系列课程(11)
查看>>
在可编辑div中插入文字或图片的问题解决思路
查看>>
发现问题,是解决问题的第一步
查看>>
c#中在规定时间弹出窗体
查看>>
win7高级搜索
查看>>
Asp.net Ajax框架教程
查看>>
C#Winform控件随窗体缩放
查看>>
JS常用代码收集
查看>>
Windows 如何在cmd命令行中查看、修改、删除与添加环境变量
查看>>