shareding分表oracle,Sharding-JDBC教程:Spring Boot整合Sharding-JDBC实现数据分表+读写分离...

点击上方“方志朋”,选择“置顶公众号”

技术文章第一时间送达!

读写分离

在上一篇文章介绍了如何使用Sharing-JDBC实现数据库的读写分离。读写分离的好处就是在并发量比较大的情况下,将查询数据库的压力 分担到多个从库中,能够满足高并发的要求。比如上一篇实现的那样,架构图如下:

44750b5102b523996177fedd92f996b7.png

数据分表

当数据量比较大的时候,比如单个表的数据量超过了500W的数据,这时可以考虑将数据存储在不同的表中。比如将user表拆分为四个表user0、user1、 user2、user3装在四个表中。此时如图所示:

d2b4134859d4fd574c2c0b71d786f507.png

案例详解

和上一篇文章使用的数据库是同一个数据库,数据库信息如下:

数据库类型

数据库

ip

cool

10.0.0.3

cool

10.0.0.13

cool

10.0.0.17

在主库初始化Mysql数据的脚本,初始化完后,从库也会创建这些表,脚本信息如下:

USE`cool`;

/*Table structure for table `user_0` */

DROP TABLE IF EXISTS`user_0`;

CREATE TABLE`user_0`(

`id`int(12)NOT NULL AUTO_INCREMENT,

`username`varchar(12)NOT NULL,

`password`varchar(30)NOT NULL,

PRIMARY KEY(`id`),

KEY`idx-username`(`username`)

)ENGINE=InnoDBAUTO_INCREMENT=149DEFAULT CHARSET=utf8;

/*Table structure for table `user_1` */

DROP TABLE IF EXISTS`user_1`;

CREATE TABLE`user_1`(

`id`int(12)NOT NULL AUTO_INCREMENT,

`username`varchar(12)NOT NULL,

`password`varchar(30)NOT NULL,

PRIMARY KEY(`id`),

KEY`idx-username`(`username`)

)ENGINE=InnoDBAUTO_INCREMENT=150DEFAULT CHARSET=utf8;

/*Table structure for table `user_2` */

DROP TABLE IF EXISTS`user_2`;

CREATE TABLE`user_2`(

`id`int(12)NOT NULL AUTO_INCREMENT,

`username`varchar(12)NOT NULL,

`password`varchar(30)NOT NULL,

PRIMARY KEY(`id`),

KEY`idx-username`(`username`)

)ENGINE=InnoDBAUTO_INCREMENT=147DEFAULT CHARSET=utf8;

/*Table structure for table `user_3` */

DROP TABLE IF EXISTS`user_3`;

CREATE TABLE`user_3`(

`id`int(12)NOT NULL AUTO_INCREMENT,

`username`varchar(12)NOT NULL,

`password`varchar(30)NOT NULL,

PRIMARY KEY(`id`),

KEY`idx-username`(`username`)

)ENGINE=InnoDBAUTO_INCREMENT=148DEFAULT CHARSET=utf8;

本案例还是在上一篇文章的案例基础之上进行改造,工程的目录和pom的依赖见上一篇文章或者源码。在工程的配置 文件application.yml做Sharding-JDBC的配置,代码如下:

sharding:

jdbc:

dataSource:

names:db-test0,db-test1,db-test2

db-test0:#org.apache.tomcat.jdbc.pool.DataSource

type:com.alibaba.druid.pool.DruidDataSource

driverClassName:com.mysql.jdbc.Driver

url:jdbc:mysql://10.0.0.3:3306/cool?useUnicode=true&characterEncoding=utf8&tinyInt1isBit=false&useSSL=false&serverTimezone=GMT

username:root

password:

maxPoolSize:20

db-test1:

type:com.alibaba.druid.pool.DruidDataSource

driverClassName:com.mysql.jdbc.Driver

url:jdbc:mysql://10.0.0.13:3306/cool?useUnicode=true&characterEncoding=UTF-8&allowMultiQueries=true&useSSL=false&serverTimezone=GMT

username:root

password:

maxPoolSize:20

db-test2:

type:com.alibaba.druid.pool.DruidDataSource

driverClassName:com.mysql.jdbc.Driver

url:jdbc:mysql://10.0.0.17:3306/cool?useUnicode=true&characterEncoding=UTF-8&allowMultiQueries=true&useSSL=false&serverTimezone=GMT

username:root

password:

maxPoolSize:20

props:

sql:

show:true

sharding.jdbc.config.sharding.tables.user.actual-data-nodes:ds_0.user_$->{0..3}

sharding.jdbc.config.sharding.tables.user.table-strategy.standard.sharding-column:id

sharding.jdbc.config.sharding.tables.user.table-strategy.standard.precise-algorithm-class-name:com.forezp.sharedingjdbcmasterslavetables.MyPreciseShardingAlgorithm

sharding.jdbc.config.sharding.master-slave-rules.ds_0.master-data-source-name:db-test0

sharding.jdbc.config.sharding.master-slave-rules.ds_0.slave-data-source-names:db-test1,db-test2

在上面的配置中,sharding.jdbc.dataSource部分是配置的数据源的信息,本案例有三个数据源db-test0、db-test1、db-test2。

sharding.jdbc.config.sharding.master-slave-rules.ds0.master-data-source-name配置的是主库的数据库名,本案例为db-test0,其中ds0为分区名。

sharding.jdbc.config.sharding.master-slave-rules.ds_0.slave-data-source-names配置的是从库的数据库名,本案例为db-test1、db-test2。

sharding.jdbc.config.sharding.tables.user.actual-data-nodes配置的分表信息,真实的数据库信息。ds0.user$->{0..3},表示读取ds0数据源的user0、user1、user2、user_3。

sharding.jdbc.config.sharding.tables.user.table-strategy.standard.sharding-column配置的数据分表的字段,是根据id来分的。

sharding.jdbc.config.sharding.tables.user.table-strategy.standard.precise-algorithm-class-name是配置数据分表的策略的类,这里是自定义的类MyPreciseShardingAlgorithm。

MyPreciseShardingAlgorithm是根据id取模4来获取表名的,代码如下:

publicclassMyPreciseShardingAlgorithmimplementsPreciseShardingAlgorithm{

@Override

publicStringdoSharding(CollectionavailableTargetNames,PreciseShardingValueshardingValue){

for(StringtableName:availableTargetNames){

if(tableName.endsWith(shardingValue.getValue()%4+"")){

returntableName;

}

}

thrownewIllegalArgumentException();

}

}

测试

写一个API来测试,代码如下:

@RestController

publicclassUserController{

Loggerlogger=LoggerFactory.getLogger(UserController.class);

@Autowired

privateUserServiceuserService;

@GetMapping("/users")

publicObjectlist(){

returnuserService.list();

}

@GetMapping("/add")

publicObjectadd(){

for(inti=100;i<150;i++){

Useruser=newUser();

user.setId(i);

user.setUsername("forezp"+(i));

user.setPassword("1233edwd");

longresutl=userService.addUser(user);

logger.info("insert:"+user.toString()+" result:"+resutl);

}

return"ok";

}

}

启动Spring Boot工程,在浏览器上执行localhost:8080/add,然后去数据库中查询,可以看到user0、user1、user2、user3分别插入了数据。 然后访问localhost:8080/users,可以查询数据库中四个表中的所有数据。可见Sharding-JDBC在插入数据的时候,根据数据分表策略,将数据存储在 不同的表中,查询的时候将数据库从多个表中查询并聚合。

在数据库的主机的日志里面,可以看到查询的日志也验证了这个结论,如下:

2019-06-20T02:50:25.183174Z2030Queryselect@@session.transaction_read_only

2019-06-20T02:50:25.193506Z2030QueryINSERT INTO user_2(

id,username,password

)

VALUES(

134,

'forezp134',

'1233edwd'

)

...省略

从库查询日志:

2019-06-20T02:41:28.450643Z7367QuerySELECT u.*FROM user_1 u

2019-06-20T02:41:28.450644Z7366QuerySELECT u.*FROM user_0 u

2019-06-20T02:41:28.461238Z7367QuerySELECT u.*FROM user_3 u

2019-06-20T02:41:28.462188Z7366QuerySELECT u.*FROM user_2 u

源码

https://github.com/forezp/SpringBootLearning/tree/master/sharding-jdbc-example/shareding-jdbc-master-slave-tables

参考资料

https://github.com/apache/incubator-shardingsphere-example/releases/tag/3.1.0.M1

https://shardingsphere.apache.org/document/current/cn/overview/

https://github.com/apache/incubator-shardingsphere

https://mp.weixin.qq.com/s/VlJ3oN0Us2eZPk0sDT7w

热门内容:

感谢搓一下“在看

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值