基础概念
ShardingSphere-JDBC是Apache ShardingSphere项目中的一个子项目,Apache ShardingSphere是一款分布式的数据库生态系统,可以通过分片、弹性伸缩、加密等能力对原有数据库进行增强。
ShardingSphere-JDBC定位是轻量级Java框架,在JDBC层提供额外服务。它能尽量透明化水平分库分表所带来的影响,让业务方逻辑上感知到一个数据库节点和逻辑表,当收到SQL查询,主要做了以下工作:
- SQL解析:词法解析和语法解析,提炼出解析上下文
- SQL路由:根据解析上下文匹配用户配置的库表的分片策略,并生成路由后的SQL(一条或多条)。
- SQL改写:将SQL改写为物理数据库能正常执行的语句(逻辑SQL -> 物理SQL)。
- SQL执行:通过多线程异步执行改写后的SQL语句。
- 结果归并:将多个执行结果归并为统一的JDBC接口输出。
几个概念:
-
逻辑表:ORM框架的业务层面,表现为一张表,例如:t_order
-
物理表:数据库层面实际存在的表,例如:t_order_0、t_order_1
-
绑定表:分片规则一致的一组分片表,进行关联查询时,建议使用分片键进行关联,否则影响查询效率。
SELECT i.* FROM t_order o JOIN t_order_item i ON o.order_id=i.order_id WHERE o.order_id in (10, 11);
-
广播表:数据源中都存在的表,且结构和数据都完全一致。适用于数据量不大且需要与其他大数据量表进行关联查询。
-
分片键:根据某个字段的计算结果(取模等)进行水平分片
-
路由:通过SQL语句中的信息,找到对应分片的过程
分片算法
-
标准分片算法:单一键作为分片键。
-
取模算法:根据一些字段,或多个字段hash求值再取模。
-
范围限定算法,按照年份、实践等策略路由到目标数据库。
-
-
复合分片算法:多键作为分片键,自行设计
-
Hint分片算法:用于处理使用Hint行分片的场景(非数据库字段的分片方式)
分片策略
包含分片键和分片算法,ShardingSphere-JDBC提供了以下几种分片策略:
- 标准分片策略(StandardSharingStrategy):使用精确分片算法或范围分片算法,支持单分片键。
- 复合分片策略(ComplexShardingStrategy):使用复合分片算法,支持多分片键。
- Hint分片策略(HintShardingStrategy):使用Hint分片算法
- Inline分片策略(InlineShardingStrategy):使用groovy表达式作为分片算法
- 不分片策略(NoneShardingStrategy):不使用分片算法
Demo
pom.xml
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>2.1.4</version>
</dependency>
<dependency>
<groupId>com.mysql</groupId>
<artifactId>mysql-connector-j</artifactId>
<scope>runtime</scope>
<version>8.0.32</version>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
<dependency>
<groupId>com.baomidou</groupId>
<artifactId>mybatis-plus-boot-starter</artifactId>
<version>3.2.0</version>
</dependency>
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>1.2.15</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.apache.shardingsphere</groupId>
<artifactId>sharding-jdbc-spring-boot-starter</artifactId>
<version>4.0.0-RC1</version>
</dependency>
<dependency>
<groupId>org.apache.shardingsphere</groupId>
<artifactId>sharding-core-common</artifactId>
<version>4.0.0-RC1</version>
</dependency>
ShardingSphere提供了多种配置方式:
- Java代码配置
- yaml、properties配置
- Spring Boot配置
该案例使用配置文件方式:application.properties
spring.shardingsphere.datasource.names=test-0,test-1
spring.main.allow-bean-definition-overriding=true
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
spring.shardingsphere.datasource.test-0.type=com.alibaba.druid.pool.DruidDataSource
spring.shardingsphere.datasource.test-0.driver-class-name=com.mysql.jdbc.Driver
spring.shardingsphere.datasource.test-0.url=jdbc:mysql://172.27.184.50:3306/test-0?useUnicode=true&characterEncoding=utf8&tinyInt1isBit=false&useSSL=false
spring.shardingsphere.datasource.test-0.username=root
spring.shardingsphere.datasource.test-0.password=123456
spring.shardingsphere.datasource.test-1.type=com.alibaba.druid.pool.DruidDataSource
spring.shardingsphere.datasource.test-1.driver-class-name=com.mysql.jdbc.Driver
spring.shardingsphere.datasource.test-1.url=jdbc:mysql://172.27.184.50:3306/test-1?useUnicode=true&characterEncoding=utf8&tinyInt1isBit=false&useSSL=false
spring.shardingsphere.datasource.test-1.username=root
spring.shardingsphere.datasource.test-1.password=123456
spring.shardingsphere.sharding.default-database-strategy.inline.sharding-column=order_id
spring.shardingsphere.sharding.default-database-strategy.inline.algorithm-expression=test-$->{order_id % 2}
spring.shardingsphere.sharding.tables.t_order.actual-data-nodes=test-$->{0..1}.t_order_$->{0..2}
spring.shardingsphere.sharding.tables.t_order.table-strategy.inline.sharding-column=order_id
spring.shardingsphere.sharding.tables.t_order.table-strategy.inline.algorithm-expression=t_order_$->{order_id % 3}
spring.shardingsphere.sharding.broadcast-tables=t_config
spring.shardingsphere.props.sql.show=true
分别创建test-0
和test-1
数据库,并创建以下表:
CREATE TABLE `t_order_0` (
`order_id` bigint(200) NOT NULL,
`order_no` varchar(100) DEFAULT NULL,
`create_name` varchar(50) DEFAULT NULL,
`price` decimal(10,2) DEFAULT NULL,
PRIMARY KEY (`order_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC;
CREATE TABLE `t_order_1` (
`order_id` bigint(200) NOT NULL,
`order_no` varchar(100) DEFAULT NULL,
`create_name` varchar(50) DEFAULT NULL,
`price` decimal(10,2) DEFAULT NULL,
PRIMARY KEY (`order_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC;
CREATE TABLE `t_order_2` (
`order_id` bigint(200) NOT NULL,
`order_no` varchar(100) DEFAULT NULL,
`create_name` varchar(50) DEFAULT NULL,
`price` decimal(10,2) DEFAULT NULL,
PRIMARY KEY (`order_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC;
CREATE TABLE `t_config` (
`id` bigint(30) NOT NULL,
`remark` varchar(50) CHARACTER SET utf8 DEFAULT NULL,
`create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
`last_modify_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=latin1;
@Data
@TableName("t_order")
@ToString
public class TOrder {
private Long orderId;
private String orderNo;
private String createName;
private BigDecimal price;
}
@Service
public class OrderService {
@Resource
private OrderMapper orderMapper;
private static final AtomicLong ID = new AtomicLong(1);
public void insertOrder() {
for (int i = 0; i < 1000; i++) {
TOrder order = new TOrder();
order.setOrderId(ID.getAndIncrement());
order.setOrderNo("A000" + i);
order.setCreateName("订单 " + i);
order.setPrice(new BigDecimal("" + i));
orderMapper.insert(order);
}
}
public List<TOrder> selectList(){
QueryWrapper<TOrder> queryWrapper = new QueryWrapper<>();
queryWrapper.like("create_name", "订单");
return orderMapper.selectList(queryWrapper);
}
}
@Mapper
public interface OrderMapper extends BaseMapper<TOrder> {
}
@Component
@Slf4j
public class Runner implements ApplicationRunner {
@Resource
private OrderService orderService;
@Override
public void run(ApplicationArguments args) throws Exception {
orderService.insertOrder();
// List<TOrder> orderList = orderService.selectList();
// log.info("orderList : {}", orderList);
}
}
查询数据库,数据按照指定规则存储: