shardingJDBC按月分表,可否实现动态建立表,而后进行分表?

如题。html

如今有个任务,一张大表数据上亿,须要按照月份拆分。用shardingJDBC的技术进行分表。java

这个是log数据,并且会一直增长,mysql

也就是说,即便如今你建立了2018年12个月,2019年12个月,等到2020年一月份怎么办?linux

难道要一开始建立好几年的表,而后等过了这些年,再手动建立新表而后上代码改配置吗?git

全网没有搜到解决方案,github

到看到一个跟我同样诉求的帖子:if shardingsphere can create tables automatically? #2521 里面并无解决办法web

还有一个说是无法动态建立表:Sharding-JDBC的动态分表spring

 

但好像都没明确说明为何不能动态分表,大概看了一下分表规则核心代码:sql

wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

分表设置须要在TableRule上设置具体分表有那几个表:actualTables(tnames)数据库

那我是否是能够在分片的时候,假如我按照goods_id%4分片,我如今已经在数据库里建立了goods_0,goods_1两张表,当goods_id%4==2或者3的时候,用JDBC建立表,而后让shardingJDBC知道我新建了一个表,分片规则里的表就多一个,不就行了嘛。

 

【我分析认为:

一旦建立了DataSource,就分配了分表规则,已经写死在里面了,

新建一个表,表名是否是得从新写一个分表规则,新建一个buildDataSource()?

这些都没用吧?由于人家shardingJDBC就认你一开始制定的规则,后面加的不认识。。】

 

若是有大佬实现了这个功能,求告诉我。。。。

 

如今来看看我是怎么掉坑里的,想把问题抛出来,看看有没有解决办法,

没有的话,那就只能手动建个20年的表(不知道会不会被打死)

 

如今来看看个人demo怎么实现的:

架构用的是springboot + Maven + JPA + MySQL +sharding-jbdc,

感兴趣的同窗能够本身按照个人目录从新构建项目,代码我不放github了。

 

这个demo除了动态建表分表,没有成功以外, (不须要这部分代码能够本身去掉,等下告诉你删哪些代码)

它很好的实现了:(分库的部分我注释掉了,感兴趣的能够本身完善它,网上也有相关代码)

1. 按id奇偶分表;

2. 分布式自增主键;

3. 测试分表save,update,select,delete;

4. 测试查询跨表数据;

5. 测试查询指定id list数据;

6. 用java建立MySQL 数据表;

 

目录是: (我把无关的目录划掉了)

wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

pom.xml :

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
	<modelVersion>4.0.0</modelVersion>
	<parent>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-parent</artifactId>
		<version>2.1.3.RELEASE</version>
		<relativePath/> <!-- lookup parent from repository -->
	</parent>
	<groupId>com.example.ziqiiii</groupId>
	<artifactId>sharding-jdbc-demo</artifactId>
	<version>0.0.1-SNAPSHOT</version>
	<name>sharding-jdbc-demo</name>
	<description>Demo project for Spring Boot</description>

	<properties>
		<java.version>1.8</java.version>
	</properties>

	<dependencies>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-data-jpa</artifactId>
		</dependency>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-web</artifactId>
		</dependency>

		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-devtools</artifactId>
			<scope>runtime</scope>
		</dependency>
		<dependency>
			<groupId>mysql</groupId>
			<artifactId>mysql-connector-java</artifactId>
			<scope>runtime</scope>
		</dependency>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-test</artifactId>
			<scope>test</scope>
		</dependency>
		<!-- lombok -->
		<dependency>
			<groupId>org.projectlombok</groupId>
			<artifactId>lombok</artifactId>
			<optional>true</optional>
		</dependency>
		<!-- druid -->
		<dependency>
			<groupId>com.alibaba</groupId>
			<artifactId>druid</artifactId>
			<version>1.1.9</version>
		</dependency>
		<!-- sharding-jdbc -->
		<dependency>
			<groupId>com.dangdang</groupId>
			<artifactId>sharding-jdbc-core</artifactId>
			<version>1.5.4</version>
		</dependency>

		<dependency>
			<groupId>org.jadira.usertype</groupId>
			<artifactId>usertype.core</artifactId>
			<version>7.0.0.CR1</version>
		</dependency>
		<dependency>
			<groupId>org.jadira.usertype</groupId>
			<artifactId>usertype.jodatime</artifactId>
			<version>2.0.1</version>
		</dependency>
		<dependency>
			<groupId>joda-time</groupId>
			<artifactId>joda-time-hibernate</artifactId>
			<version>1.4</version>
		</dependency>
		<dependency>
			<groupId>joda-time</groupId>
			<artifactId>joda-time</artifactId>
			<version>2.9.9</version>
		</dependency>

	</dependencies>

	<build>
		<plugins>
			<plugin>
				<groupId>org.springframework.boot</groupId>
				<artifactId>spring-boot-maven-plugin</artifactId>
			</plugin>
		</plugins>
	</build>

</project>
wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

config下面的TableShardingAlgorithm.java

package com.example.ziqiiii.shardingjdbcdemo.config;

import com.dangdang.ddframe.rdb.sharding.api.ShardingValue;
import com.dangdang.ddframe.rdb.sharding.api.strategy.table.SingleKeyTableShardingAlgorithm;
import com.example.ziqiiii.shardingjdbcdemo.database.DataSourceConfig;
import com.google.common.collect.Range;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import java.sql.Connection;
import java.sql.Statement;
import java.util.Collection;
import java.util.LinkedHashSet;
import java.util.List;

/**
 * create by ziqi on 2019/8/19
 */
@Component
public class TableShardingAlgorithm implements SingleKeyTableShardingAlgorithm<Long> {
    private final String tablePrefix = "goods_";

    @Autowired
    private DataSourceConfig dataSourceConfig;


    @Override
    public String doEqualSharding( Collection<String> tableNames,  ShardingValue<Long> shardingValue) {

        //1.
        //分表:goods_0,goods_1:
//       String endStr = shardingValue.getValue() % 2 + "";
//       String tb = tablePrefix + endStr;
//       return tb;

        
        //下面的代码是动态建立表,并进行分表
        //建立表功能OK,可是分表报错,不须要动态建立分表的话,能够注释掉下面部分
        //2.
        //动态建立表,假设分片规则为id%4,已经建立了0,1表,那么它会自动建立2,3表
        List<String> databsesTable = dataSourceConfig.getTnames();

        String endStr = shardingValue.getValue() % 4 + "";
        String tb = tablePrefix + endStr;

        if(tableNames.contains(tb)){
            System.out.println(tb);
        }else{

            String creatsql = String.format("CREATE TABLE `%s` (" +
                    "  `goods_id` bigint(20) NOT NULL," +
                    "  `goods_name` varchar(100) COLLATE utf8_bin NOT NULL," +
                    "  `goods_type` bigint(20) DEFAULT NULL," +
                    "  PRIMARY KEY (`goods_id`)" +
                    ") ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin",tb);


            Connection conn = dataSourceConfig.getConnection();
            //执行建立表
            System.out.println("//建立表");
            try{
                Statement stmt = conn.createStatement();
                if(0 == stmt.executeUpdate(creatsql)) {
                    System.out.println("成功建立表!");
                } else {
                    System.out.println("建立表失败!");
                }
                //
                stmt.close();
                conn.close();
                System.out.println("//关闭资源");

                databsesTable.add(tb);
                tableNames.add(tb);

            }catch (Exception e){
                e.printStackTrace();
            }


        }

        return tb;


    }

    @Override
    public Collection<String> doInSharding(final Collection<String> tableNames, final ShardingValue<Long> shardingValue) {
        Collection<String> result = new LinkedHashSet<>(tableNames.size());
        for (Long value : shardingValue.getValues()) {
            for (String tableName : tableNames) {
                if (tableName.endsWith(value % 2 + "")) {
                    result.add(tableName);
                }
            }
        }
        return result;
    }

    @Override
    public Collection<String> doBetweenSharding(final Collection<String> tableNames,
                                                final ShardingValue<Long> shardingValue) {
        Collection<String> result = new LinkedHashSet<>(tableNames.size());
        Range<Long> range = shardingValue.getValueRange();
        for (Long i = range.lowerEndpoint(); i <= range.upperEndpoint(); i++) {
            for (String each : tableNames) {
                if (each.endsWith(i % 2 + "")) {
                    result.add(each);
                }
            }
        }
        return result;
    }
}
wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

controller下面的GoodsController.java

package com.example.ziqiiii.shardingjdbcdemo.controller;

import com.dangdang.ddframe.rdb.sharding.keygen.DefaultKeyGenerator;
import com.example.ziqiiii.shardingjdbcdemo.entity.Goods;
import com.example.ziqiiii.shardingjdbcdemo.repository.GoodsRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.ArrayList;
import java.util.List;

/**
 * create by ziqi on 2019/8/19
 */
@RestController
public class GoodsController {

    @Autowired
    private DefaultKeyGenerator keyGenerator;

    @Autowired
    private GoodsRepository goodsRepository;

    @GetMapping("save")
    public String save(){
        for(int i= 11 ; i <= 20 ; i ++){

            Goods goods = new Goods();
            Number n = keyGenerator.generateKey();
//            System.out.println("keyGenerator:"+n);
//            goods.setGoodsId((long) n);
            goods.setGoodsId((long) i);
            goods.setGoodsName( "shi" + i);
            goods.setGoodsType((long) (i+1));
            goodsRepository.save(goods);
        }
        return "success";
    }

    @GetMapping("update")
    public String update(){
        for(int i= 1 ; i <= 10 ; i ++){
//            Goods goods = new Goods();
//            goods.setGoodsId((long) i);
//            goods.setGoodsName( "ziqi" + i);
//            goods.setGoodsType((long) (i+1));
            int res = goodsRepository.update((long) i,"ziqi" + i);
        }
        return "success";
    }

    @GetMapping("select")
    public String select(){
        return goodsRepository.findAll().toString();
    }

    @GetMapping("delete")
    public void delete(){
        goodsRepository.deleteAll();
    }

    @GetMapping("query1")
    public Object query1(){
        return goodsRepository.findAllByGoodsIdBetween(10L, 30L);
    }

    @GetMapping("query2")
    public Object query2(){
        List<Long> goodsIds = new ArrayList<>();
        goodsIds.add(10L);
        goodsIds.add(15L);
        goodsIds.add(20L);
        goodsIds.add(25L);
        return goodsRepository.findAllByGoodsIdIn(goodsIds);
    }

}
wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

database下面的Database0Config.java (这部分是分库的一个库)

package com.example.ziqiiii.shardingjdbcdemo.database;

import com.alibaba.druid.pool.DruidDataSource;
import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;

import javax.sql.DataSource;

/**
 * create by ziqi on 2019/8/16
 */
@Data
@ConfigurationProperties(prefix = "database0")
@Component
public class Database0Config {
    private String url;
    private String username;
    private String password;
    private String driverClassName;
    private String databaseName;

    public DataSource createDataSource() {
        DruidDataSource result = new DruidDataSource();
        result.setDriverClassName(getDriverClassName());
        result.setUrl(getUrl());
        result.setUsername(getUsername());
        result.setPassword(getPassword());
        return result;
    }
}
wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==
Database1Config.java (这部分是分库的另外一个库)须要分库的这个注释能够去掉
package com.example.ziqiiii.shardingjdbcdemo.database;

/**
 * create by ziqi.zhang on 2019/8/19
 */
//@Data
//@ConfigurationProperties(prefix = "database1")
//@Component
//public class Database1Config {
//    private String url;
//    private String username;
//    private String password;
//    private String driverClassName;
//    private String databaseName;
//
//    public DataSource createDataSource() {
//        DruidDataSource result = new DruidDataSource();
//        result.setDriverClassName(getDriverClassName());
//        result.setUrl(getUrl());
//        result.setUsername(getUsername());
//        result.setPassword(getPassword());
//        return result;
//    }
//}
wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==
DataSourceConfig.java
package com.example.ziqiiii.shardingjdbcdemo.database;

/**
 * create by ziqi on 2019/8/16
 */

import com.dangdang.ddframe.rdb.sharding.api.ShardingDataSourceFactory;
import com.dangdang.ddframe.rdb.sharding.api.rule.DataSourceRule;
import com.dangdang.ddframe.rdb.sharding.api.rule.ShardingRule;
import com.dangdang.ddframe.rdb.sharding.api.rule.TableRule;
import com.dangdang.ddframe.rdb.sharding.api.strategy.table.TableShardingStrategy;
import com.dangdang.ddframe.rdb.sharding.keygen.DefaultKeyGenerator;

import com.example.ziqiiii.shardingjdbcdemo.config.TableShardingAlgorithm;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import javax.sql.DataSource;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.*;


@Configuration
public class DataSourceConfig {


    private List<String> tnames = new ArrayList<>();
    //构造代码块,在构造函数调用前调用

    {
        tnames.add("goods_0");
        tnames.add("goods_1");
    }

    @Autowired
    private Database0Config database0Config;

    @Autowired
    private TableShardingAlgorithm tableShardingAlgorithm;



    @Bean
    public DataSource getDataSource() throws SQLException {
        return buildDataSource();
    }


    @Bean
    public List<String> getTnames(){
        return tnames;
    }

    public boolean setTnames(String name){
        return tnames.add(name);
    }

    /*
    * 获取sql链接
    */
    public Connection getConnection() {
        //原生JDBC开发
        DataSource dataSource = null;
        Connection conn = null;
        try {
            dataSource = getDataSource();
            conn = dataSource.getConnection();
            if (!conn.isClosed()) {
                System.out.println("数据库链接成功!");
            }

        } catch (SQLException e) {
            e.printStackTrace();
        }

        return conn;

    }

    private DataSource buildDataSource() throws SQLException {
        //分库设置
        Map<String, DataSource> dataSourceMap = new HashMap<>(1);
        //添加两个数据库database0和database1
        dataSourceMap.put(database0Config.getDatabaseName(), database0Config.createDataSource());
//        dataSourceMap.put(database1Config.getDatabaseName(), database1Config.createDataSource());

        //设置默认数据库
        DataSourceRule dataSourceRule = new DataSourceRule(dataSourceMap, database0Config.getDatabaseName());


        //分表设置,大体思想就是将查询虚拟表Goods根据必定规则映射到真实表中去
        //Goods的分表规则
        TableRule orderTableRule = TableRule.builder("goods")
//                .actualTables(Arrays.asList("goods_0", "goods_1")) //这样写
                .actualTables(tnames)   //或者这样
                .dataSourceRule(dataSourceRule)
                .build();
        //分表设置,history_按照月份划分
//        List<String> historyTables = Arrays.asList("history_0", "history_1","history_2",
//                "history_3","history_4","history_5","history_6",
//                "history_7","history_8","history_9","history_10",
//                "history11");
//
//        TableRule orderTableRule = TableRule.builder("history_wti_alarm")
//                .actualTables(historyTables)
//                .dataSourceRule(dataSourceRule)
//                .build();

       

        //分库分表策略
        ShardingRule shardingRule = ShardingRule.builder()
                .dataSourceRule(dataSourceRule)
                .tableRules(Arrays.asList(orderTableRule))
                .tableShardingStrategy(new TableShardingStrategy("goods_id", tableShardingAlgorithm)).build();

        DataSource dataSource = ShardingDataSourceFactory.createDataSource(shardingRule);
        return dataSource;
    }


    @Bean
    public DefaultKeyGenerator keyGenerator() {
        DefaultKeyGenerator kg = new DefaultKeyGenerator();

        InetAddress address;
        Long workerId;
        try {
            address = InetAddress.getLocalHost();
        } catch (final UnknownHostException e) {
            throw new IllegalStateException("Cannot get LocalHost InetAddress, please check your network!");
        }
        // 先获得服务器的hostname,例如JTCRTVDRA44,linux上可经过命令"cat /proc/sys/kernel/hostname"查看;
        String hostName = address.getHostName();

        try {
            // 计算workerId的方式:
            // 第一步hostName.replaceAll("\\d+$", ""),即去掉hostname后纯数字部分,例如JTCRTVDRA44去掉后就是JTCRTVDRA
            // 第二步hostName.replace(第一步的结果, ""),即将原hostname的非数字部分去掉,获得纯数字部分,就是workerId
            // workerId = Long.valueOf(hostName.replace(hostName.replaceAll("\\d+$", ""), ""));
            String tmp = hostName.replaceAll("\\d+$", "");
            String tmp2 = hostName.replace(tmp, "");
            if(("").equals(tmp2)){
                workerId = 0L;
            }else {
                workerId = Long.valueOf(tmp2);
            }
        } catch (final NumberFormatException e) {
            throw new IllegalArgumentException(String.format("Wrong hostname:%s, hostname must be end with number!", hostName));
        }

        kg.setWorkerId(workerId);
        return kg;
    }

}
wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

特意去看了一下它的.class文件,长这样:

wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

 

entity下面的Goods.java

package com.example.ziqiiii.shardingjdbcdemo.entity;

import lombok.Data;

import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.Table;

/**
 * create by ziqi on 2019/8/19
 */
@Entity
@Table(name="goods")
@Data
public class Goods {
    @Id
    private Long goodsId;

    private String goodsName;

    private Long goodsType;
}

repository下面的GoodsRepository.java

package com.example.ziqiiii.shardingjdbcdemo.repository;

import com.example.ziqiiii.shardingjdbcdemo.entity.Goods;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Modifying;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.query.Param;

import javax.transaction.Transactional;
import java.util.List;

/**
 * create by ziqi on 2019/8/19
 */
public interface GoodsRepository extends JpaRepository<Goods, Long> {

    List<Goods> findAllByGoodsIdBetween(Long goodsId1, Long goodsId2);

    List<Goods> findAllByGoodsIdIn(List<Long> goodsIds);

    @Modifying
    @Transactional
    @Query("update Goods g set g.goodsName =:name where g.goodsId =:id")
    int update(@Param("id") Long id, @Param("name")String name);
}
package com.example.ziqiiii.shardingjdbcdemo;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class ShardingJdbcDemoApplication {

	public static void main(String[] args) {
		SpringApplication.run(ShardingJdbcDemoApplication.class, args);
	}

}

 

resources下面的application.properties

##Jpa配置
spring.jpa.database=mysql
spring.jpa.show-sql=true
spring.jpa.hibernate.ddl-auto=none

##数据库配置
##数据库database0地址
database0.url=jdbc:mysql://localhost:3306/database0?characterEncoding=utf8&useSSL=false
##数据库database0用户名
database0.username=root
##数据库database0密码
database0.password=
##数据库database0驱动
database0.driverClassName=com.mysql.cj.jdbc.Driver
##数据库database0名称
database0.databaseName=database0

##数据库database1地址
database1.url=jdbc:mysql://localhost:3306/database1?characterEncoding=utf8&useSSL=false]
##数据库database1用户名
database1.username=root
##数据库database1密码
database1.password=
##数据库database1驱动
database1.driverClassName=com.mysql.jdbc.Driver
##数据库database1名称
database1.databaseName=database1

 

建表语句:

CREATE DATABASE database0;
USE database0;
DROP TABLE IF EXISTS `goods_0`;
CREATE TABLE `goods_0` (
  `goods_id` bigint(20) NOT NULL,
  `goods_name` varchar(100) COLLATE utf8_bin NOT NULL,
  `goods_type` bigint(20) DEFAULT NULL,
  PRIMARY KEY (`goods_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin;
DROP TABLE IF EXISTS `goods_1`;
CREATE TABLE `goods_1` (
  `goods_id` bigint(20) NOT NULL,
  `goods_name` varchar(100) COLLATE utf8_bin NOT NULL,
  `goods_type` bigint(20) DEFAULT NULL,
  PRIMARY KEY (`goods_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin;
CREATE DATABASE database1;
USE database1;
DROP TABLE IF EXISTS `goods_0`;
CREATE TABLE `goods_0` (
  `goods_id` bigint(20) NOT NULL,
  `goods_name` varchar(100) COLLATE utf8_bin NOT NULL,
  `goods_type` bigint(20) DEFAULT NULL,
  PRIMARY KEY (`goods_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin;

DROP TABLE IF EXISTS `goods_1`;
CREATE TABLE `goods_1` (
  `goods_id` bigint(20) NOT NULL,
  `goods_name` varchar(100) COLLATE utf8_bin NOT NULL,
  `goods_type` bigint(20) DEFAULT NULL,
  PRIMARY KEY (`goods_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin;