一、实战
1、引入 ShardingSphere-JDBC 的依赖
https://mvnrepository.com/artifact/org.apache.shardingsphere/shardingsphere-jdbc/5.5.0
XML
<!-- https://mvnrepository.com/artifact/org.apache.shardingsphere/shardingsphere-jdbc -->
<dependency>
<groupId>org.apache.shardingsphere</groupId>
<artifactId>shardingsphere-jdbc</artifactId>
<version>5.5.0</version>
<exclusions>
<exclusion>
<groupId>org.apache.shardingsphere</groupId>
<artifactId>shardingsphere-test-util</artifactId>
</exclusion>
</exclusions>
</dependency>
下面是我项目的一些版本信息:
SpringBoot 3.2.4
shardingsphere 5.5.0
data:image/s3,"s3://crabby-images/0ac41/0ac4134bd11742272372277a9e4fa881b52af9d3" alt=""
2、修改yaml配置文件
application.yaml:
XML
spring:
profiles:
active: dev
main:
allow-circular-references: true
datasource:
# ShardingSphere 对 Driver 自定义,实现分库分表等隐藏逻辑
driver-class-name: org.apache.shardingsphere.driver.ShardingSphereDriver
# ShardingSphere 配置文件路径
url: jdbc:shardingsphere:classpath:shardingsphere-config.yaml
# driver-class-name: ${quick.datasource.driver-class-name}
# url: jdbc:mysql://${quick.datasource.host}:${quick.datasource.port}/${quick.datasource.database}?serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=utf-8&zeroDateTimeBehavior=convertToNull&useSSL=false&allowPublicKeyRetrieval=true&rewriteBatchedStatements=true
# username: ${quick.datasource.username}
# password: ${quick.datasource.password}
# hikari:
# maximum-pool-size: 50
data:image/s3,"s3://crabby-images/6abe2/6abe2504add3aee686f1968c46454615f7ee23b4" alt=""
shardingsphere-config.yaml
data:image/s3,"s3://crabby-images/00404/00404bf3251d8a7245865357ae864e20e56969ff" alt=""
名字必须相同,前面的 jdbc:shardingsphere:classpath: 是固定写法。
data:image/s3,"s3://crabby-images/e253b/e253b5c914974c0e8750ca62ec89d27fde707f07" alt=""
大致的配置结构如上。
XML
dataSources:
ds_0:
dataSourceClassName: com.zaxxer.hikari.HikariDataSource
driverClassName: com.mysql.cj.jdbc.Driver
jdbcUrl: jdbc:mysql://127.0.0.1:3306/quick_pickup?serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=utf-8&zeroDateTimeBehavior=convertToNull&useSSL=false&allowPublicKeyRetrieval=true&rewriteBatchedStatements=true
username: root
password: 123456
# 规则配置
rules:
- !SINGLE
tables:
# 加载全部单表
- "ds_0.*"
- !SHARDING
tables:
integral_package_order:
actualDataNodes: ds_0.integral_package_order_${0..15} # 配置数据表分片规则
tableStrategy:
standard:
shardingColumn: user_id # 使用 user_id 作为分片键
shardingAlgorithmName: integral_package_order_table_hash_mod # 使用自定义的分片算法
shardingAlgorithms:
integral_package_order_table_hash_mod:
# type: HASH_MOD # 使用 HASH_MOD 算法
# props:
# sharding-count: 16 # 设置分片数量为 16
type: INLINE
props:
algorithm-expression: integral_package_order_${user_id % 16}
props:
sql-show: true # 控制台打印 SQL 日志,便于调试
# 单机模式 也有集群模式 此处本机选择单机即可 这个必须添加! 不然要报错
mode:
type: Standalone
其中注意必须加上:
data:image/s3,"s3://crabby-images/fff41/fff413b6b3ca5a38f5249cf7878dad714bb9bdda" alt=""
就是两点可能错误:
1.xxx表无法加载
Caused by: org.apache.ibatis.executor.ExecutorException: Error preparing statement. Cause: org.apache.shardingsphere.infra.exception.kernel.metadata.TableNotFoundException: Table or view 'store' does not exist. at org.apache.ibatis.executor.statement.BaseStatementHandler.prepare(BaseStatementHandler.java:99) ~[mybatis-3.5.14.jar:3.5.14] at org.apache.ibatis.executor.statement.RoutingStatementHandler.prepare(RoutingStatementHandler.java:60) ~[mybatis-3.5.14.jar:3.5.14]
2.不能使用自动的算法这类型的报错
Caused by: com.zaxxer.hikari.pool.HikariPool$PoolInitializationException: Failed to initialize pool: Sharding algorithm
HASH_MOD
initialization failed, reason is:integral_package_order
tables sharding configuration can not use auto sharding algorithm.. at com.zaxxer.hikari.pool.HikariPool.throwPoolInitializationException(HikariPool.java:596) ~[HikariCP-5.0.1.jar:na]
上面截图中HashMod的分片数量里面这个integral_package_order_${order_id % 16}属性应该是16才对,但是还是报错,有大佬有解决可以评论区教教。
设置好需要分片的表,下面就是对表进行水平分片
3、对表进行水平分片
data:image/s3,"s3://crabby-images/efb3b/efb3bacc5ea694ea6fb90097da5e14b83cadb623" alt=""
这里我写了一个测试案例可以帮忙我生成16个建表语句:
data:image/s3,"s3://crabby-images/55f83/55f83b5dac7fa1a97267bac6209f52c8106bfada" alt=""
java
package com.quick.sharding;
public class IntegralPackageOrderShardingTest {
// 基础 SQL 模板,使用占位符 `%d` 替换分表的后缀
public static final String SQL_TEMPLATE = "create table integral_package_order_%d\n" +
"(\n" +
" id bigint not null comment '主键'\n" +
" primary key,\n" +
" user_id bigint unsigned not null comment '抢购积分包的用户id',\n" +
" integral_package_id bigint unsigned not null comment '抢购的积分包id',\n" +
" create_time timestamp default CURRENT_TIMESTAMP not null comment '抢购时间',\n" +
" has_use int null comment '是否使用(0未使用 1已使用)'\n" +
")\n" +
" comment '积分包订单' charset = utf8mb4\n" +
" row_format = COMPACT;\n" +
"\n" +
"create index idx_userId_integral_package_order_%d\n" +
" on integral_package_order_%d (user_id);\n";
public static void main(String[] args) {
// 生成 16 个分表 SQL
for (int i = 0; i < 16; i++) {
// 替换占位符并打印结果
String sql = String.format(SQL_TEMPLATE, i, i, i);
System.out.println(sql);
}
}
}
输出如下:
data:image/s3,"s3://crabby-images/f2931/f29317a849c5a381952fcfdeafb31d9c2337a19c" alt=""
data:image/s3,"s3://crabby-images/1fd52/1fd52f4eaa2f10a45ed45854d417ad3e8e96622a" alt=""
如何运行这些sql即可:
data:image/s3,"s3://crabby-images/75a4e/75a4ef8689c90b471dd4489fdca34b446098f69a" alt=""
分表完成后理论上是全部配置完成了,但是你会发现运行了没报错,但是发现接口测试,发现了会出现后面的一些问题,这里我先给出解决的配置,就是在你的 WebMvcConfiguration 中加上下面那一段配置。
java
/**
* 参考:<a href="https://hafuhafu.com/archives/springboot-jackson-xml-result-json/">...</a>
* 设置默认的内容类型,并禁用检查Accept请求头,服务端就会忽视原有请求的Accept中的内容协商类型,而按照配置的默认的类型进行响应。
* 按以下配置的情况下,请求默认总是返回JSON。
* 如果想让部分接口只返回XML格式的数据,可以在@RequestMapping中配置produces,
* 如@GetMapping(value = "/api/users",produces = MediaType.APPLICATION_XHTML_XML_VALUE),那个接口就会返回XML格式的数据。
* @param configurer 配置器
*/
@Override
public void configureContentNegotiation(ContentNegotiationConfigurer configurer) {
configurer
.defaultContentType(MediaType.APPLICATION_JSON, MediaType.APPLICATION_XML, MediaType.APPLICATION_XHTML_XML)
.ignoreAcceptHeader(true);
}
data:image/s3,"s3://crabby-images/87374/87374236c5daaa010919a4890b00f132460ef6e3" alt=""
4、解决问题
这里只是讲述第三点最后部分解决问题的一些方案,如果不想深入了解可以直接看测试分表那边。
data:image/s3,"s3://crabby-images/61294/612943f621bcb6d0717d959f250f5ab5e30a6ee9" alt=""
傻眼了,为什么是xml格式,也就是下面这类型的问题:
Spring 5 -jackson-dataformat-xml forces @ResponseBody with XML
结果是XML格式,而不是我们希望的Json格式,我们之前配置的Json序列化失效了吗,这个是我当时的怀疑,其实这个也还是好解决的,只困扰了我一小会,如何去搜,下面先给出解决办法。
data:image/s3,"s3://crabby-images/2f83c/2f83cd61467e50864d86586a29a64670c4b95fb2" alt=""
我在 WebMvcConfiguration 加了这个一个配置,然后运行解决了
data:image/s3,"s3://crabby-images/5a788/5a78820da273d0bc97d9cf00c92c8e5d886aaae1" alt=""
成功输出了Json格式的返回。
下面是我去Github搜的一个解决方案:
data:image/s3,"s3://crabby-images/beb60/beb60c765b2a51fd9724ed5572c93f39ba530d30" alt=""
这个大佬遇到的问题和我的一样,下面就有很多好友有做评论区解决,大概都聚集在5.5.0版本的shardingsphere让WebMVC的Json转换器失效,因为他自带了一个jackson-dataformat-xml这个东西
data:image/s3,"s3://crabby-images/d2d3f/d2d3f9621c45624810806afe6c8242bc5ef3591f" alt=""
data:image/s3,"s3://crabby-images/abdc4/abdc413a0325cbc235f86c42b4fb35c79b565df6" alt=""
这个大佬有提到移除那个依赖,我下面试了这样子做:
data:image/s3,"s3://crabby-images/5d19c/5d19c9c4baca203b8abd8c80bc35c5dc128004d5" alt=""
结果却报了另外的错误。。。可能是我没完全理解意思,还有另外的策略。
data:image/s3,"s3://crabby-images/e291b/e291b25e65f63b3db2974a1664789ab792480578" alt=""
最后我采用了这个策略,再加一个配置
data:image/s3,"s3://crabby-images/52d2d/52d2d6dac00dabbd6de034e2eaf307f58243262d" alt=""
阅读了那个大佬多年工作经验的解决办法,成功加了一个配置解决了那个问题,但是在我的项目里面就多了两个Json序列化,有点在shi山上又多加了几行的感觉。。。
data:image/s3,"s3://crabby-images/2f219/2f21918ee9d3ad7564394f8b4485b477da11e03b" alt=""
但是还是解决了,但是发现接口文档出现下面这个问题
data:image/s3,"s3://crabby-images/7cec3/7cec31c7966c981ed0b4d0ad0e7e6f22c5770c3f" alt=""
我们主打的就是一个遇到问题解决问题,再次改造,上面的问题可能是优先级比接口文档高导致接口文档失效,这里就不调整优先级,换了另外一个解决方案,如果不使用接口文档那样子还是可以的。
将上面的配置换成:
java
/**
* 参考:<a href="https://hafuhafu.com/archives/springboot-jackson-xml-result-json/">...</a>
* 设置默认的内容类型,并禁用检查Accept请求头,服务端就会忽视原有请求的Accept中的内容协商类型,而按照配置的默认的类型进行响应。
* 按以下配置的情况下,请求默认总是返回JSON。
* 如果想让部分接口只返回XML格式的数据,可以在@RequestMapping中配置produces,
* 如@GetMapping(value = "/api/users",produces = MediaType.APPLICATION_XHTML_XML_VALUE),那个接口就会返回XML格式的数据。
* @param configurer 配置器
*/
@Override
public void configureContentNegotiation(ContentNegotiationConfigurer configurer) {
configurer
.defaultContentType(MediaType.APPLICATION_JSON, MediaType.APPLICATION_XML, MediaType.APPLICATION_XHTML_XML)
.ignoreAcceptHeader(true);
}
data:image/s3,"s3://crabby-images/4caf1/4caf1dbe771bc17e551b8911a87a706988e6ca87" alt=""
参考了这个文档:内容协商:SpringBoot添加jackson-xml后浏览器返回值的变化 -- 败犬不需要安可
mark一下大佬的解决方案。
data:image/s3,"s3://crabby-images/44b5f/44b5feddb673abffd62dea06a719a8e702ea6554" alt=""
5、测试分表
这里从测试结合我之前做过的这些博客食用更佳:
基于之前的秒杀功能的优化(包括Sentinel在SpringBoot中的简单应用)-CSDN博客
1.添加积分包
data:image/s3,"s3://crabby-images/6b191/6b19175d46d05b6c42fa0a6da5bf5be9d5c501d9" alt=""
data:image/s3,"s3://crabby-images/9de67/9de67c5082c0db55f67377b5d1016d72beb7d74e" alt=""
这里已经是可以看到ShardingSphere生效了。
2.查询所有积分包
data:image/s3,"s3://crabby-images/165a9/165a9b6edcbb23d48e714093f2d3b6b6bb374902" alt=""
3.启用禁用积分包
data:image/s3,"s3://crabby-images/d15d9/d15d9cba7214aefd679681397c05d88a9c73f9b0" alt=""
redis中如下:
data:image/s3,"s3://crabby-images/1cfa9/1cfa99aa302b7e8b613515ed689b91f2400a0809" alt=""
可以看到库存已经存在,为100.
4.用户查看所有秒杀积分包信息
data:image/s3,"s3://crabby-images/bea51/bea510b242dd0d57f435355e7f7134b9b3584647" alt=""
5.根据积分包id查询库存
data:image/s3,"s3://crabby-images/cd8c7/cd8c7f4948ffcc85085fa82b678859e9211e38dc" alt=""
6.用户秒杀抢购积分包
data:image/s3,"s3://crabby-images/b3895/b3895059d4e1a769d68d6ceb00719f9e5d4bd815" alt=""
data:image/s3,"s3://crabby-images/dcaf1/dcaf102682a5b75acce23ccfc0c1558d6eaf0cf0" alt=""
在数据库中可以看出已经存在一条数据了
data:image/s3,"s3://crabby-images/b2829/b282927f55989eab459d4f5654c558b6cef59505" alt=""
redis中库存也-1
data:image/s3,"s3://crabby-images/c1c14/c1c142603374ebc2edf2dfeba3f3499a24579f35" alt=""
订单信息也存在
7.用户查看自己的积分包订单
data:image/s3,"s3://crabby-images/db49c/db49c963bc64200c6572fb3f5ed8e9b994581042" alt=""
8.禁用积分包
data:image/s3,"s3://crabby-images/50d18/50d1852ec498e8c466d145d001102ac435c9572b" alt=""
data:image/s3,"s3://crabby-images/fa129/fa129788ad410ee09e195c212a60eeeb9cd1b767" alt=""
9.删除积分包
data:image/s3,"s3://crabby-images/2c34d/2c34d5e6d6041973b3d25719d249a5acb0682bd1" alt=""
data:image/s3,"s3://crabby-images/dd334/dd334b026c6cdac4cdd77f5c7cbfac8345590ba6" alt=""
删除成功
10.压测
由于上诉是连接服务器的数据库进行测试,下面用本地数据库进行压测
jmeter配置:
data:image/s3,"s3://crabby-images/dbda8/dbda8d71646cfdc0a8ea01b34dd4d0a62a8243ee" alt=""
data:image/s3,"s3://crabby-images/6646c/6646c8f7f1a9cfeb9d88ea25aea91c4b0aba8add" alt=""
data:image/s3,"s3://crabby-images/20723/20723eec542b5642cb3eaf8b09c7def615dae2d2" alt=""
data:image/s3,"s3://crabby-images/2764a/2764a03332bb6b669f7d74cd88f37f3aa75062ca" alt=""
data:image/s3,"s3://crabby-images/d4947/d4947dabfd40021ef02b89e9e93a71356ced1884" alt=""
开始压测:
data:image/s3,"s3://crabby-images/f73bf/f73bfd7d41edd0a1c4dca0108c068f7ed2e46486" alt=""
如下可见,已经分别分布在各个表上,分表成功:
data:image/s3,"s3://crabby-images/22611/226110ea02303bd35c03cce167efc80fec8ac07c" alt=""
data:image/s3,"s3://crabby-images/92aed/92aedd247813b565625cd9501dde79b314aaace6" alt=""
data:image/s3,"s3://crabby-images/1b2f8/1b2f8bd104c7b16a640f28720c8afe4dec8826bb" alt=""
二、还有一些遇到过的bug
之前用的是下面这个依赖,遇到各种形形色色的问题,可能是版本不服的问题吧,如果有大佬知道怎么解决可以评论区说说,还有上面为什么不能使用HashMod这个算法的原因......
java
<!-- https://mvnrepository.com/artifact/org.apache.shardingsphere/shardingsphere-jdbc-core -->
<dependency>
<groupId>org.apache.shardingsphere</groupId>
<artifactId>shardingsphere-jdbc-core</artifactId>
<version>5.3.2</version>
</dependency>