动态表名 的使用方法

动态表名插件的底层是 拦截器

1,创建一个拦截器


@Configuration
public class MybatisConfiguration {

    @Bean
    public DynamicTableNameInnerInterceptor dynamicTableNameInnerInterceptor() {
        // 准备一个Map,用于存储TableNameHandler
        Map<String, TableNameHandler> map = new HashMap<>(1);
        // 存入一个TableNameHandler,用来替换points_board表名称
        // 替换方式,就是从TableInfoContext中读取保存好的动态表名
        map.put("points_board", (sql, tableName) -> TableInfoContext.getInfo() == null ? tableName : TableInfoContext.getInfo());
        return new DynamicTableNameInnerInterceptor(map);
    }
}

拦截器创建好了并不会生效,还需要配置到mybatisPlus的配置中:


@Configuration
@ConditionalOnClass({MybatisPlusInterceptor.class, BaseMapper.class})
public class MybatisConfig {

    /**
     * @deprecated 存在任务更新数据导致updater写入0或null的问题,暂时废弃
     * @see MyBatisAutoFillInterceptor 通过自定义拦截器来实现自动注入creater和updater
     */
    // @Bean
    // @ConditionalOnMissingBean
    public BaseMetaObjectHandler baseMetaObjectHandler(){
        return new BaseMetaObjectHandler();
    }

    //配置mybatis plus 的拦截器链
    //DynamicTableNameInnerInterceptor 不是所用服务都用到 ,目前只有tj-learning服务用到,
    // 因此加上@Autowared(required = false)注解 注入时声明非必须
    @Bean
    @ConditionalOnMissingBean
    public MybatisPlusInterceptor mybatisPlusInterceptor(@Autowired(required = false) DynamicTableNameInnerInterceptor dynamicTableNameInnerInterceptor) {
        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();

        if (dynamicTableNameInnerInterceptor != null) {
            interceptor.addInnerInterceptor(dynamicTableNameInnerInterceptor);//动态表名拦截器插件
        }

        PaginationInnerInterceptor paginationInnerInterceptor = new PaginationInnerInterceptor(DbType.MYSQL);
        paginationInnerInterceptor.setMaxLimit(200L);
        interceptor.addInnerInterceptor(paginationInnerInterceptor);//分页拦截器插件
        interceptor.addInnerInterceptor(new MyBatisAutoFillInterceptor());//自动填充拦截器插件

        return interceptor;
    }
}

该如何传递表名称呢?

从计算表名,到动态表名插件执行,调用TableNameHandler,都是在一个线程内完成的。要在一个线程内实现数据共享,该用什么呢?

就是ThreadLocal.

我们可以在定时任务中计算完动态表名后,将表名存入ThreadLocal,然后在插件中从ThreadLocal中读取即可:

定义一个传递表名称的工具,具体代码如下:

package com.tianji.learning.utils;

public class TableInfoContext {
    private static final ThreadLocal<String> TL = new ThreadLocal<>();

    public static void setInfo(String info) {
        TL.set(info);
    }

    public static String getInfo() {
        return TL.get();
    }

    public static void remove() {
        TL.remove();
    }
}

相关推荐

  1. mybatis-plus 动态简易使用

    2024-05-14 13:28:08       40 阅读
  2. MyBatis Plus中动态实践

    2024-05-14 13:28:08       40 阅读
  3. django根据时间(年月日)动态修改--方法

    2024-05-14 13:28:08       41 阅读
  4. elementUI 动态校验单数据方法

    2024-05-14 13:28:08       46 阅读
  5. SpringBoot MybatisPlus 配置动态&多数据源

    2024-05-14 13:28:08       22 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-05-14 13:28:08       94 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-05-14 13:28:08       100 阅读
  3. 在Django里面运行非项目文件

    2024-05-14 13:28:08       82 阅读
  4. Python语言-面向对象

    2024-05-14 13:28:08       91 阅读

热门阅读

  1. vue2 el-tree树形下拉框

    2024-05-14 13:28:08       25 阅读
  2. RateLimiter 限流算法使用

    2024-05-14 13:28:08       27 阅读
  3. Python学习-Numpy-2

    2024-05-14 13:28:08       31 阅读
  4. Android自由控制阴影位置

    2024-05-14 13:28:08       31 阅读
  5. Vue setup函数

    2024-05-14 13:28:08       32 阅读
  6. Kotlin标准函数和静态方法

    2024-05-14 13:28:08       31 阅读