由于公司的项目运行越久,数据量就越大,一般来说,mysql达到1000万数据就要考虑分表了,这里我们使用MybatisPlus拦截器实现业务分表。
1.MySQL初步设定按时间分表半年分一次表,一年两张表,要么存储过程实现自动创建表,或者手工创建一年两张表
2.如果是实现一年两张表,则需新建辅助表,主要三个字段,表名,开始时间,结束时间,根据辅助表找到分表后的表名,一般配合前端传过来的时间查询使用
3.MybatisPlus 拦截器设置动态表名,在SQL执行前,替换成分表后的表名
package com.yourcom.proname.config;
import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
import com.baomidou.mybatisplus.extension.plugins.PaginationInterceptor;
import com.baomidou.mybatisplus.extension.plugins.handler.TableNameHandler;
import com.baomidou.mybatisplus.extension.plugins.inner.DynamicTableNameInnerInterceptor;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.util.StringUtils;
import java.util.HashMap;
/**
* mybatisPlus动态表解析器
*
*/
@Configuration
@MapperScan("com.yourcom.proname.repository.mapper")
public class MybatisPlusConfig {
// 需要替换的动态表名
public static ThreadLocal<String> TABLE_NAME = new ThreadLocal<>();
@Bean
public MybatisPlusInterceptor mybatisPlusInterceptor(){
MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
DynamicTableNameInnerInterceptor dynamicTableNameInnerInterceptor = new DynamicTableNameInnerInterceptor();
HashMap<String, TableNameHandler> map = new HashMap<>();
// 对于merchant表,进行动态表名设置
map.put("merchant", (sql, tableName) -> {
String table=TABLE_NAME.get();
TABLE_NAME.remove();
// 返回动态表名
return !StringUtils.isEmpty(table) ? tableName + "_" + table : tableName;
});
dynamicTableNameInnerInterceptor.setTableNameHandlerMap(map);
interceptor.addInnerInterceptor(dynamicTableNameInnerInterceptor);
return interceptor;
}
@Bean
public PaginationInterceptor paginationInterceptor() {
PaginationInterceptor paginationInterceptor = new PaginationInterceptor();
// 设置sql的limit为无限制,默认是500
paginationInterceptor.setLimit(-1);
return paginationInterceptor;
}
}
4.如下是业务层使用,添加后则替换原表去执行动态表名
@Override
public List<Merchant> queryMerchantConnectTable(Map<String, Object> param) {
String [] strTimeArr = param.get("startTime").toString().trim().split("-");
//此处添加表名拦截,为具体表名后缀
MybatisPlusConfig.TABLE_NAME.set(strTimeArr[0]+strTimeArr[1]);
return merchantMapper.queryMerchantConnectTable(param);
}