• shardingJdbc分库分表实战


    Apache ShardingSphere 是一款分布式的数据库生态系统, 可以将任意数据库转换为分布式数据库,并通过数据分片、弹性伸缩、加密等能力对原有数据库进行增强。

    Apache ShardingSphere 设计哲学为 Database Plus,旨在构建异构数据库上层的标准和生态。 它关注如何充分合理地利用数据库的计算和存储能力,而并非实现一个全新的数据库。 它站在数据库的上层视角,关注它们之间的协作多于数据库自身。

    ShardingSphere-JDBC 定位为轻量级 Java 框架,在 Java 的 JDBC 层提供的额外服务。 它使用客户端直连数据库,以 jar 包形式提供服务,无需额外部署和依赖,可理解为增强版的 JDBC 驱动,完全兼容 JDBC 和各种 ORM 框架。

    • 适用于任何基于 JDBC 的 ORM 框架,如:JPA, Hibernate, Mybatis, Spring JDBC Template 或直接使用 JDBC;
    • 支持任何第三方的数据库连接池,如:DBCP, C3P0, BoneCP, HikariCP 等;
    • 支持任意实现 JDBC 规范的数据库,目前支持 MySQL,PostgreSQL,Oracle,SQLServer 以及任何可使用 JDBC 访问的数据库。

    下面讲解一下shardingJdbc的实战部分

    1.分表操作

    话说,我现在有个课程表,需要按照cid基数和偶数放在不同的表中。当然实际使用中规则,可能不一样。表的数量也不一样,这里举一个简单的例子。

    两个表的表结构如下

    1. CREATE TABLE `course_1` (
    2. `cid` bigint(20) NOT NULL,
    3. `cname` varchar(50) NOT NULL,
    4. `user_id` bigint(20) NOT NULL,
    5. `cstatus` varchar(10) NOT NULL,
    6. PRIMARY KEY (`cid`)
    7. ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
    1. CREATE TABLE `course_2` (
    2. `cid` bigint(20) NOT NULL,
    3. `cname` varchar(50) NOT NULL,
    4. `user_id` bigint(20) NOT NULL,
    5. `cstatus` varchar(10) NOT NULL,
    6. PRIMARY KEY (`cid`)
    7. ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

    在pom.xml引入依赖

    1. <dependency>
    2. <groupId>mysql</groupId>
    3. <artifactId>mysql-connector-java</artifactId>
    4. <scope>runtime</scope>
    5. </dependency>
    6. <dependency>
    7. <groupId>org.apache.shardingsphere</groupId>
    8. <artifactId>sharding-jdbc-spring-boot-starter</artifactId>
    9. <version>4.0.0-RC1</version>
    10. </dependency>
    11. <dependency>
    12. <groupId>org.mybatis.spring.boot</groupId>
    13. <artifactId>mybatis-spring-boot-starter</artifactId>
    14. <version>2.2.2</version>
    15. </dependency>

    application.properties中配置规则数据源和规则

    这里我们只是需要连接一个数据库,所以只要配置一个数据源就可以了。

    1. spring.shardingsphere.datasource.names =m1
    2. spring.shardingsphere.datasource.m1.type = com.alibaba.druid.pool.DruidDataSource
    3. spring.shardingsphere.datasource.m1.driver-class-name=com.mysql.jdbc.Driver
    4. spring.shardingsphere.datasource.m1.url=jdbc:mysql://localhost:3306/course_db?serverTimerzone=GMT%2B8
    5. spring.shardingsphere.datasource.m1.username=root
    6. spring.shardingsphere.datasource.m1.password=abc
    7. #配置表的规则 m1.course_1,m1.course_2
    8. spring.shardingsphere.rules.sharding.tables.course.actual-data-nodes=m1.course_$->{1..2}
    9. #配置主键字段
    10. spring.shardingsphere.sharding.tables.course.key-generator.column=cid
    11. #主键字段的生成规则,这里是雪花算法
    12. spring.shardingsphere.sharding.tables.course.key-generator.type=SNOWFLAKE
    13. spring.main.allow-bean-definition-overriding=true
    14. #配置分片策略的字段
    15. spring.shardingsphere.sharding.tables.course.table-strategy.inline.sharding-column=cid
    16. spring.shardingsphere.sharding.tables.course.table-strategy.inline.algorithm-expression=course_$->{cid % 2 +1}
    17. #打印sql
    18. spring.shardingsphere.props.sql.show=true

    这里使用mybatis作为数据持久层框架。

    1. @Mapper
    2. public interface CourseMapper {
    3. void insertCourse(Course c);
    4. }
    1. mapper
    2. PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
    3. "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
    4. <mapper namespace="com.keelon.zsimall.usedmall.mapper.CourseMapper">
    5. <insert id="insertCourse" parameterType="com.keelon.zsimall.usedmall.entity.Course">
    6. insert into course (cid,cname,user_id,cstatus) values (#{cid}, #{cname},#{userId},#{cstatus})
    7. insert>
    8. <select id="queryCoursee" resultType="com.keelon.zsimall.usedmall.entity.Course">
    9. select * from course where cid = #{cid} and user_id =#{userId}
    10. select>
    11. mapper>

    实体类

    1. package com.keelon.zsimall.usedmall.entity
    2. class Course {
    3. private Long cid;
    4. private String cname;
    5. private Long userId;
    6. private String cstatus;
    7. Long getCid() {
    8. return cid
    9. }
    10. void setCid(Long cid) {
    11. this.cid = cid
    12. }
    13. String getCname() {
    14. return cname
    15. }
    16. void setCname(String cname) {
    17. this.cname = cname
    18. }
    19. Long getUserId() {
    20. return userId
    21. }
    22. void setUserId(Long userId) {
    23. this.userId = userId
    24. }
    25. String getCstatus() {
    26. return cstatus
    27. }
    28. void setCstatus(String cstatus) {
    29. this.cstatus = cstatus
    30. }
    31. @Override
    32. public String toString() {
    33. return "Course{" +
    34. "cid=" + cid +
    35. ", cname='" + cname + '\'' +
    36. ", userId=" + userId +
    37. ", cstatus='" + cstatus + '\'' +
    38. '}';
    39. }
    40. }
    1. @SpringBootTest
    2. class UsedMallApplicationTests {
    3. @Autowired
    4. private CourseMapper courseMapper;
    5. @Test
    6. void testData() {
    7. Course course = new Course();
    8. course.setCname("zhangsan");
    9. course.setUserId(100L);
    10. course.setCid(5L);
    11. course.setCstatus("Nomarl");
    12. courseMapper.insertCourse(course);
    13. }
    14. }

    通过执行测试数据库,就可以将数据按照我们配置的落表到2个表中。

    2.分库分表

    我现在有个需求,需要根据user_id这个字段,插入到2个库中,然后再根据cid这个字段,落在两个库中。这个时候我就需要2个数据源了。这个时候,我们需要再配置上做一些改变了。

    1. spring.shardingsphere.datasource.names =m1,m2
    2. #配置数据源1
    3. spring.shardingsphere.datasource.m1.type = com.alibaba.druid.pool.DruidDataSource
    4. spring.shardingsphere.datasource.m1.driver-class-name=com.mysql.jdbc.Driver
    5. spring.shardingsphere.datasource.m1.url=jdbc:mysql://localhost:3306/edu_db_1?serverTimerzone=GMT%2B8
    6. spring.shardingsphere.datasource.m1.username=root
    7. spring.shardingsphere.datasource.m1.password=abc
    8. #配置数据源2
    9. spring.shardingsphere.datasource.m2.type = com.alibaba.druid.pool.DruidDataSource
    10. spring.shardingsphere.datasource.m2.driver-class-name=com.mysql.jdbc.Driver
    11. spring.shardingsphere.datasource.m2.url=jdbc:mysql://localhost:3306/edu_db_2?serverTimerzone=GMT%2B8
    12. spring.shardingsphere.datasource.m2.username=root
    13. spring.shardingsphere.datasource.m2.password=abc
    14. #配置数据有几个。这里是m1和m2 m1.course_1,m1.course_2
    15. spring.shardingsphere.rules.sharding.tables.course.actual-data-nodes=m$->{1..2}.course_$->{1..2}
    16. spring.shardingsphere.sharding.tables.course.key-generator.column=cid
    17. spring.shardingsphere.sharding.tables.course.key-generator.type=SNOWFLAKE
    18. #配置数据分片字段,按照那个字段作为规则插入对应的表
    19. spring.shardingsphere.sharding.tables.course.table-strategy.inline.sharding-column=cid
    20. spring.shardingsphere.sharding.tables.course.table-strategy.inline.algorithm-expression=course_$->{cid % 2 +1}
    21. #配置数据源分片的字段,根据user_id选择使用哪个数据源
    22. spring.shardingsphere.sharding.tables.course.database-strategy.inline..sharding-column=user_id
    23. spring.shardingsphere.sharding.tables.course.database-strategy.inline.algorithm-expression=m$->{user_id % 2 +1}
    24. spring.main.allow-bean-definition-overriding=true
    25. #配置sql输出打印
    26. spring.shardingsphere.props.sql.show=true

    测试类

    1. @SpringBootTest
    2. class UsedMallApplicationTests {
    3. @Autowired
    4. private CourseMapper courseMapper;
    5. @Test
    6. void tableData() {
    7. Course course = new Course();
    8. course.setCname("zhangsan");
    9. course.setUserId(100L);
    10. course.setCid(21L);
    11. course.setCstatus("Nomarl");
    12. courseMapper.insertCourse(course);
    13. Course course1 = new Course();
    14. course1.setCname("zhangsan");
    15. course1.setUserId(101L);
    16. course1.setCid(20L);
    17. course1.setCstatus("Nomarl");
    18. courseMapper.insertCourse(course1);
    19. }
    20. @Test
    21. void queryData() {
    22. Map map = new HashMap();
    23. map.put("cid",20);
    24. map.put("userId",101L);
    25. Course course = courseMapper.queryCoursee(map);
    26. System.out.println(course.toString());
    27. }
    28. }

    总体来说,我们实现分库还是分表插入的数据,对于mybatis的持久层是没有区别的,只是在配置文件有所差异,对于分库操作, 我们需要配置多个数据源,shardingjdbc根据我们配置去选择我们需要的数据源。

  • 相关阅读:
    【RabbitMQ】消息队列需要解决的几个问题
    JVM性能调优
    前端面试题---事件循环机制和异步编程
    ASTM标准涵盖哪些产品类别?出口美国的产品做ASTM认证需要注意哪些事项?
    深度学习与CV教程(13) | 目标检测 (SSD,YOLO系列)
    Mysql tinyint(1)与tinyint(4)的区别
    [前端]Preprocessor dependency "less" not found.
    HTML5+css3课后习题【一】
    学习编程的目的
    【软考:系统集成项目管理】之 项目干系人管理
  • 原文地址:https://blog.csdn.net/qq_31927785/article/details/126888971