sharding-jdbc 使用实例

2019-10-27  本文已影响0人  爱编程的凯哥

问题:

通过java编程式方式(非配置文件)完整示例,通过sharding-jdbc实现:
1. 分库,根据user_ip进行简单的分库示例
2. 分表,实现针对时间的分表
3. 读写分离,进行主从集群配置

表结构:

image.png

本次测试一共建了4个库:master1、master2、master1_1、master2_2,其中两对主从库.

  1. master1主写库----->master1_1 从读库
  2. master2主写库------> master2_2从写库
    其中测试表auth_user_log进行分表测试,看下这个表结构:
CREATE TABLE if not exists
  master1.auth_user_log
(
  id bigint NOT NULL AUTO_INCREMENT,
  log_id VARCHAR(64),
  account VARCHAR(64),
  account_name VARCHAR(64) NOT NULL,
  application_name VARCHAR(64) NOT NULL,
  user_ip VARCHAR(32) NOT NULL,
  url VARCHAR(64) NOT NULL,
  request_body VARCHAR(2048) NOT NULL,
  reponse_body VARCHAR(2048) NOT NULL,
  invoke_time TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
  invoke_cost INT,
  PRIMARY KEY (id),
  CONSTRAINT log_id UNIQUE (log_id)
  )
  ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='用户访问记录表';

我们通过

  1. master1和master2 两个主库来测试根据user_ip字段进行分库操作,
  2. 根据invoke_time进行时间的分表测试
  3. 根据master1和master1_1进行读写分离配置测试

源代码路径

https://gitee.com/kaiyang_taichi/demo-shariding-jdbc.git

数据库脚本:

image.png

配置讲解:

  1. 通过springboot配置了4个库的数据源master1, master2, master1_1, master2_2
  2. 通过MasterSlaveRuleConfiguration进行读写分离配置,可以配置多从库,选择从库的负载策略,对于读多写少的业务场景很实用
  3. 然后针对单个表的配置通过TableRuleConfiguration进行处理.
  4. 根据user_ip的hashcode进行分库,通过UserLogDbShardingAlgorithm类进行处理
  5. 根据invoke_time字段进行分表,通过 UserLogTableShardingAlgorithm和UserLogTableRangeShardingAlgorithm(查询一段时间范围内的数据使用)配置.
@Configuration
public class ShardingDataSourceConfiguration {


    /**
     * 主写库1
     * @return
     */
    @Bean("master1")
    @ConfigurationProperties(prefix = "spring.datasource.master1")
    public DataSource dataSource1() {
        return DataSourceBuilder.create().build();
    }

    /**
     * 主写库2
     * @return
     */
    @Bean("master2")
    @ConfigurationProperties(prefix = "spring.datasource.master2")
    public DataSource dataSource2() {
        return DataSourceBuilder.create().build();
    }

    /**
     * 从读库1
     * @return
     */
    @Bean("master1_1")
    @ConfigurationProperties(prefix = "spring.datasource.master11")
    public DataSource dataSource1_1() {
        return DataSourceBuilder.create().build();
    }

    /**
     * 从读库2
     * @return
     */
    @Bean("master2_2")
    @ConfigurationProperties(prefix = "spring.datasource.master22")
    public DataSource dataSource2_2() {
        return DataSourceBuilder.create().build();
    }



    @Bean(name = "shardDataSource")
    @Primary
    public DataSource shardDataSource(@Qualifier("master1") DataSource dataSource,
        @Qualifier("master2") DataSource dataSource2,@Qualifier("master1_1") DataSource dataSource3,
        @Qualifier("master2_2") DataSource dataSource4) throws SQLException {

        ShardingRuleConfiguration shardingRuleConfig = new ShardingRuleConfiguration();

        List<MasterSlaveRuleConfiguration> configurationList = new ArrayList<>();

        //1.读写分离配置一,master1为写库1,master1_1为读库1,这两个库将会在数据库层面进行配置主从数据同步
        MasterSlaveRuleConfiguration masterSlaveRuleConfiguration = new MasterSlaveRuleConfiguration();
       // 一组主从库分组
        masterSlaveRuleConfiguration.setName("master1Config");
      //主库名
        masterSlaveRuleConfiguration.setMasterDataSourceName("master1");
        List<String> slaves = new ArrayList<>();
        slaves.add("master1_1");
    //从库列表
        masterSlaveRuleConfiguration.setSlaveDataSourceNames(slaves);
   //从库的负载选择策略        masterSlaveRuleConfiguration.setLoadBalanceAlgorithmType(MasterSlaveLoadBalanceAlgorithmType.ROUND_ROBIN);
        configurationList.add(masterSlaveRuleConfiguration);

        //2.读写分离配置二,master1为写库2,master2_2为读库2,这两个库将会在数据库层面进行配置主从数据同步
        MasterSlaveRuleConfiguration masterSlaveRuleConfiguration2 = new MasterSlaveRuleConfiguration();
        masterSlaveRuleConfiguration2.setName("master2Config");
        masterSlaveRuleConfiguration2.setMasterDataSourceName("master2");
        List<String> slaves2 = new ArrayList<>();
        slaves2.add("master2_2");
        masterSlaveRuleConfiguration2.setSlaveDataSourceNames(slaves2);
        masterSlaveRuleConfiguration2.setLoadBalanceAlgorithmType(MasterSlaveLoadBalanceAlgorithmType.ROUND_ROBIN);
        configurationList.add(masterSlaveRuleConfiguration2);

        //3,将读写分离配置加到主配置类中
        shardingRuleConfig.setMasterSlaveRuleConfigs(configurationList);

        // 4。将分库分表规则加入到主配置类中
        shardingRuleConfig.getTableRuleConfigs().add(getUserLogTableRuleConfiguration());

        // 打印SQL
        Properties props = new Properties();
        props.put("sql.show", true);

        Map<String, DataSource> map = new HashMap<>();
        map.put("master1", dataSource);
        map.put("master2", dataSource2);
        map.put("master1_1", dataSource3);
        map.put("master2_2", dataSource4);
        return new ShardingDataSource(shardingRuleConfig.build(map),
            new ConcurrentHashMap(), props);
    }


    /**
     * 配置用户日志读分库分表策略
     * 1。 根据user_ip的hashcode进行分库,写入不同主库
     *          目前主写库有两个:master1,master2
     * 2。根据invoke_time进行分表,写入不同的子表
     *          目前子表有:
     *          auth_user_log,默认表
     *          auth_user_log_201903,19年第三季度表
     *          auth_user_log_201904,19年第四季度表
     * @return
     */
    @Bean
    TableRuleConfiguration getUserLogTableRuleConfiguration() {
        TableRuleConfiguration orderTableRuleConfig = new TableRuleConfiguration();

        orderTableRuleConfig.setLogicTable("auth_user_log");
        orderTableRuleConfig.setLogicIndex("invoke_time");

        //设置数据库分库策略,根据ip的hash值,按2的倍数分库
        orderTableRuleConfig.setDatabaseShardingStrategyConfig(
            new StandardShardingStrategyConfiguration("user_ip", UserLogDbShardingAlgorithm.class.getName()));

        // 设置分表策略
        orderTableRuleConfig.setTableShardingStrategyConfig(
            new StandardShardingStrategyConfiguration("invoke_time", UserLogTableShardingAlgorithm.class.getName(),
                UserLogTableRangeShardingAlgorithm.class.getName()));

        // 设置数据节点,格式为dbxx.tablexx。这里的名称要和map的别名一致。下面两种方式都可以
        orderTableRuleConfig.setActualDataNodes(UserLogTableRangeShardingAlgorithm.toUserLogDbConfigString());

        return orderTableRuleConfig;
    }


    @Bean(name = "transactionManager")
    public DataSourceTransactionManager registerPowerTransactionManager(
        @Qualifier("shardDataSource") DataSource shardDataSource) throws Throwable {
        DataSourceTransactionManager dataSourceTransactionManager = new DataSourceTransactionManager();
        dataSourceTransactionManager.setDataSource(shardDataSource);
        return dataSourceTransactionManager;
    }

    //
    @Bean(name = "sqlSessionFactory")
    public SqlSessionFactoryBean sqlSessionFactoryBean(@Qualifier("shardDataSource") DataSource shardDataSource)
        throws Throwable {
        SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();
        sqlSessionFactoryBean.setDataSource(shardDataSource);
        //mapper
        ResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();
        Resource[] mapperResources = resolver.getResources("classpath:mapper/*Mapper.xml");
        sqlSessionFactoryBean.setMapperLocations(mapperResources);
        //mybatis-config
        ResourceLoader loader = new DefaultResourceLoader();
        Resource configLocation = loader.getResource("classpath:mybatis-config.xml");
        sqlSessionFactoryBean.setConfigLocation(configLocation);
        return sqlSessionFactoryBean;
    }

}

分库配置讲解:

  1. 简单的规则模拟
  2. 注意: 匹配配置的数据源别名,此处注意,当配置主从集群的时候,这块dbNmame是主从组的组名,如果不配置集群组时,这里配置的是两个库自己的别名,如此时没有主从的话,此处匹配master1和master2
public class UserLogDbShardingAlgorithm implements PreciseShardingAlgorithm<String> {


    private static final String DB1 = "master1Config";

    private static final String DB2 = "master2Config";


    @Override
    public String doSharding(Collection<String> availableTargetNames, PreciseShardingValue<String> shardingValue) {
        String dbNmame;
    //简单的规则模拟场景,具体算法根据实际处理
        String ip = shardingValue.getValue();
        int i = ip.hashCode();
        dbNmame = i % 2 == 0 ? DB1 : DB2;
    // 匹配配置的数据源别名,此处注意,当配置主从集群的时候,这块dbNmame是主从组的组名,如果不配置集群组时,这里配置的是两个库自己的别名,如此时没有主从的话,此处匹配master1和master2
        for (String each : availableTargetNames) {
            if (each.equals(dbNmame)) {
                return each;
            }
        }

        throw new IllegalArgumentException();
    }
}

分表规则讲解:

  1. 示例为RangeShardingAlgorithm类型配置进行范围时间段分表配置,简单equey处理逻辑类似,只是shardingValue只有一个值
  2. 我们的处理是根据传入时间的月份进行按季度匹配对应表名
  3. 且此处通过dbConfig进行配置相关所有表名,此处一样注意master1Config此处为主从集群组名,当不需要主从时,此处配置为两个主库名即可
public class UserLogTableRangeShardingAlgorithm implements RangeShardingAlgorithm<Timestamp> {


    private static final String TblUserLogName = "auth_user_log";

    public static final Map<Integer, String> dbConfig = new HashMap<>();

    static {
        dbConfig.put(0, "master1Config." + TblUserLogName+",master2Config."+TblUserLogName);
        dbConfig.put(2019, "master1Config.auth_user_log_201903,master1Config.auth_user_log_201904,master2Config.auth_user_log_201903,master2Config.auth_user_log_201904");
    }

    public static String toUserLogDbConfigString() {
        Collection<String> values = dbConfig.values();
        if (values.size() > 0) {
            StringBuilder stringBuilder = new StringBuilder();
            values.stream().forEach(value -> stringBuilder.append(value).append(","));
            return stringBuilder.substring(0, stringBuilder.length() - 1);
        }
        return TblUserLogName;
    }

    @Override
    public Collection<String> doSharding(Collection<String> collection,
        RangeShardingValue<Timestamp> shardingValue) {
        List<String> result = Arrays.asList(TblUserLogName);
        Range<Timestamp> valueRange = shardingValue.getValueRange();
        Timestamp start = valueRange.lowerEndpoint();
        Timestamp end = valueRange.upperEndpoint();

        shardingTableNames(result, start);
        shardingTableNames(result, end);

        result.retainAll(collection);

        if (CollectionUtils.isEmpty(result)) {
            throw new IllegalArgumentException();
        }

        return result;
    }

/**
*根据时间进行分季度分表配置
**/
    private void shardingTableNames(List<String> result, Timestamp start) {
        String tb_name;
        Calendar c = Calendar.getInstance();
        c.setTimeInMillis(start.getTime());
        int year = c.get(Calendar.YEAR);
        int month = c.get(Calendar.MONTH);
        if (dbConfig.keySet().contains(year)) {
            if (month >= 1 && month <= 3) {
                tb_name = TblUserLogName + "_" + year + "01";
                result.add(tb_name);
            }
            if (month >= 4 && month <= 6) {
                tb_name = TblUserLogName + "_" + year + "02";
                result.add(tb_name);
            }
            if (month >= 7 && month <= 9) {
                tb_name = TblUserLogName + "_" + year + "03";
                result.add(tb_name);
            }
            if (month >= 10 && month <= 12) {
                tb_name = TblUserLogName + "_" + year + "04";
                result.add(tb_name);
            }
        }
    }


}

总结

到此分库、分表、读写分离的具体配置说明就说完了.shariding-jdbc通过客户端在datasource层进行封装,在sql的执行过程中进行二次封装.实现分库分表规则,简单实用.不想mycat等需要运维配置代理进行维护处理.

上一篇 下一篇

猜你喜欢

热点阅读