aop+HikariCP+Springboot+Mybatis+mysql实现分库分表

架构图
在这里插入图片描述
借助SpringBoot能够快速实现分库分表。其核心模块有3个,配置中心、mybatis插件、分库分表算法

配置中心

package com.ratel.exception.config;

import com.zaxxer.hikari.HikariConfig;
import com.zaxxer.hikari.HikariDataSource;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.plugin.Interceptor;
import org.mybatis.spring.SqlSessionFactoryBean;
import org.mybatis.spring.boot.autoconfigure.SpringBootVFS;
import org.mybatis.spring.mapper.MapperScannerConfigurer;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.context.ApplicationContext;
import org.springframework.context.EnvironmentAware;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.env.Environment;
import org.springframework.stereotype.Repository;
import javax.sql.DataSource;
import java.io.IOException;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

@Configuration
public class MultiDataSourceConfig implements EnvironmentAware {

    private Map<Object, Object> targetDatasource = new ConcurrentHashMap<>();

    private Object defaultTargetDatasource = null;

    @Bean(name = "*")
    public TableSelectorPlugin plugin() {
        return new TableSelectorPlugin();
    }

    @Bean(name = "*")
    public DataSource dataSourceV2() {
        MultiDatasource multiDatasource1 = new MultiDatasource();
        multiDatasource1.setTargetDataSources(targetDatasource);
        multiDatasource1.setDefaultTargetDataSource(defaultTargetDatasource);
        return multiDatasource1;
    }


    @Bean(name = "*")
    public SqlSessionFactoryBean sqlSessionFactory(@Qualifier(value = "*") DataSource dataSource, @Qualifier(value = "*") TableSelectorPlugin tableSelectorPlugin, ApplicationContext applicationContext) throws IOException {
        SqlSessionFactoryBean factoryBean = new SqlSessionFactoryBean();
        factoryBean.setDataSource(dataSource);
        factoryBean.setVfs(SpringBootVFS.class);
        factoryBean.setPlugins(tableSelectorPlugin);
        return factoryBean;
    }

    @Bean(name = "*")
    public MapperScannerConfigurer mapperScannerConfigurer() {
        MapperScannerConfigurer mapperScannerConfigurer = new MapperScannerConfigurer();
        mapperScannerConfigurer.setSqlSessionFactoryBeanName("*");
        mapperScannerConfigurer.setBasePackage("*");
        mapperScannerConfigurer.setAnnotationClass(Mapper.class);
        return mapperScannerConfigurer;
    }


    @Override
    public void setEnvironment(Environment environment) {
        String prefix = "router.jdbc.datasource.";
        String dataSources = environment.getProperty(prefix + "list");
        String[] split = dataSources.split(",");
        for (String item : split) {
            HikariConfig config = new HikariConfig(* + item + ".properties");
            this.targetDatasource.put(item, new HikariDataSource(config));
            if (item.equals("default")) {
                this.defaultTargetDatasource = new HikariDataSource(config);
            }
        }
        System.out.println("222"+targetDatasource.keySet());
    }
}

分表插件

    public Object intercept(Invocation invocation) throws Throwable {
        Object paramObj = invocation.getArgs()[1];
        if (!(paramObj instanceof Map)) {
            return invocation.proceed();
        } else {
            Map<String, Object> params = (Map)paramObj;
            String tableName = TABLE_NAMES.get();
            if (!StringUtils.isEmpty(tableName) && !params.containsKey(TABLE_ARG_NAME)) {
                params.put(TABLE_ARG_NAME, tableName);
            }
            return invocation.proceed();
        }
    }

分表算法

package com.ratel.exception.config;

import org.springframework.stereotype.Component;

@Component
public class DsNameDeciderImp implements DsNameDecider{
    @Override
    public DsName decideDsName(String dbName, int dbCount, long dbShardedValue, String tableName, int tableCount, long tableShardedValue) {
        return new DsName(this.hashDbName(dbName, dbCount, dbShardedValue), this.hashTableName(tableName, tableCount, tableShardedValue));

    }

    @Override
    public Integer hashDB(String dbName, int dbCount, long shardedValue) {
        return null;
    }

    @Override
    public int hashTable(String tableName, int tableCount, long shardedValue) {
        return 0;
    }

    public String hashDbName(String dbName, int dbCount, long shardedValue) {
        return dbCount == 1 ? dbName : dbName + this.hashDB(dbName, dbCount, shardedValue);
    }

    public String hashTableName(String tableName, int tableCount, long shardedValue) {
        return tableCount == 1 ? tableName : tableName + this.hashTable(tableName, tableCount, shardedValue);
    }
}

以上便是分库分表的核心代码。只需要将数据源继承AbstractRoutingDataSource抽象类。便可以轻松实现分库分表的功能了

  • 8
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
实现接口调用频率限制可以使用AOP和ConcurrentHashMap结合的方式。 首先,在Spring Boot中,我们可以使用AOP来拦截接口的调用。我们可以定义一个切面,使用@Aspect注解标注,然后在切入点方法中定义需要拦截的注解。 例如,我们可以定义一个@FrequencyLimit注解,用于标注需要限制调用频率的方法: ```java @Retention(RetentionPolicy.RUNTIME) @Target({ ElementType.METHOD }) public @interface FrequencyLimit { // 限制时间段,单位为秒 int interval() default 60; // 时间段内最大请求次数 int maxCount() default 10; } ``` 然后,在切面中,我们可以拦截该注解标注的方法,并且进行限制调用频率的操作。可以使用ConcurrentHashMap来存储每个接口的调用次数和最后一次调用时间。 ```java @Component @Aspect public class FrequencyLimitAspect { private ConcurrentHashMap<String, Long> lastRequestTimeMap = new ConcurrentHashMap<>(); private ConcurrentHashMap<String, Integer> requestCountMap = new ConcurrentHashMap<>(); @Around("@annotation(frequencyLimit)") public Object frequencyLimit(ProceedingJoinPoint joinPoint, FrequencyLimit frequencyLimit) throws Throwable { Object result = null; String methodName = joinPoint.getSignature().toLongString(); long currentTime = System.currentTimeMillis(); int interval = frequencyLimit.interval(); int maxCount = frequencyLimit.maxCount(); synchronized (this) { // 获取最后一次请求时间和请求次数 Long lastRequestTime = lastRequestTimeMap.get(methodName); Integer requestCount = requestCountMap.get(methodName); if (lastRequestTime == null || currentTime - lastRequestTime >= interval * 1000) { // 如果该接口在限制时间段内没有被调用过,则重置请求次数和最后一次请求时间 lastRequestTimeMap.put(methodName, currentTime); requestCountMap.put(methodName, 1); } else { // 如果该接口在限制时间段内已经被调用过,则增加请求次数 requestCount++; if (requestCount > maxCount) { // 如果请求次数超过了限制,则抛出异常 throw new RuntimeException("Exceeded maximum request limit"); } lastRequestTimeMap.put(methodName, currentTime); requestCountMap.put(methodName, requestCount); } } // 调用原始方法 result = joinPoint.proceed(); return result; } } ``` 在切面中,我们使用synchronized关键字来保证线程安全,因为ConcurrentHashMap并不能完全保证线程安全。同时,我们使用了@Around注解来拦截被@FrequencyLimit注解标注的方法,然后在方法中实现限制调用频率的逻辑。 这样,我们就可以实现接口调用频率限制了。在需要限制调用频率的方法中,我们只需要加上@FrequencyLimit注解即可。例如: ```java @GetMapping("/test") @FrequencyLimit(interval = 60, maxCount = 10) public String test() { return "test"; } ``` 这样,每个IP地址每分钟内最多只能调用该方法10次,超过次数会抛出异常。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值