美文网首页
根据环境不同,修改表名添加数据库名

根据环境不同,修改表名添加数据库名

作者: 滴流乱转的小胖子 | 来源:发表于2022-10-01 09:45 被阅读0次

    一、前戏

    故事背景

    公司引入一套阿里云的大数据系统,汇集公司的多张表。
    正式环境使用的是大数据中间件2.0,正式环境是3.0,3.0环境查询需要 “数据库名.表名”,2.0环境查询需要“表名”。
    需要实现一个根据环境不同,修改表名的功能。

    常见的技术方案选型

    二、正文

    配置文件bootstrap.yml

     # 正式站与测试站表明对应配置文件
    # 正式站表名: 测试站表名
    dataworks:
      tableName:
        prodTableName: testDataBaseName.testTableName
    

    解析配置信息

    /**
     * @Auther: fatsnake
     * @Description":
     * @Date:2022/9/14 13:18
     * Copyright (c) 2022, zaodao All Rights Reserved.
     */
    @Configuration
    @ConfigurationProperties(prefix = "dataworks")
    public class DataBaseTableNameConfig {
    
        /**
         * 表名集合
         */
        private Map<String, String> tableName;
    
        /**
         * 构造函数
         */
        DataBaseTableNameConfig() {
        }
    
        public Map<String, String> getTableName() {
            return tableName;
        }
    
        public void setTableName(Map<String, String> tableName) {
            this.tableName = tableName;
        }
    
        @Override
        public String toString() {
            return "DataBaseTableNameConfig{"
                    + "tableName=" + tableName
                    + '}';
        }
    }
    

    mybatis插件按环境条件生效

    @Profile({"dev", "test"})
    public class FrameMyBatisPluginConfig {
    
        /**
         *
         * @param sqlSessionFactory sqlSessionFactory
         * @return String
         */
        @Bean
        public String sqlTableNameHandleInterceptor(SqlSessionFactory sqlSessionFactory) {
            //实例化插件
            SQLTableNameHandleInterceptor sqlTableNameHandleInterceptor = new SQLTableNameHandleInterceptor();
            // 为后续留好扩展
    //        //创建属性值
    //        Properties properties = new Properties();
    //        properties.setProperty("prop1","value1");
    //        //将属性值设置到插件中
    //        sqlTableNameHandleInterceptor.setProperties(properties);
            //将插件添加到SqlSessionFactory工厂
            sqlSessionFactory.getConfiguration().addInterceptor(sqlTableNameHandleInterceptor);
            return "interceptor";
        }
    }
    

    使用mybatis的StatementHandler插件,在sql执行前进行拦截

    此处使用了druid的工具类,解析表名,替换表名后重写表名

    import com.alibaba.druid.sql.SQLUtils;
    import com.alibaba.druid.sql.ast.SQLStatement;
    import com.alibaba.druid.util.JdbcConstants;

    /**
     * @Auther: fatsnake
     * @Description":
     * @Date:2022/9/14 12:47
     * Copyright (c) 2022, zaodao All Rights Reserved.
     */
    @Intercepts({@Signature(type = StatementHandler.class, method = "prepare", args = {Connection.class, Integer.class})})
    public class SQLTableNameHandleInterceptor implements Interceptor {
    
        @Override
        public Object intercept(Invocation invocation) throws Throwable {
            DataBaseTableNameConfig dataBaseTableNameConfig
                    = (DataBaseTableNameConfig)SpringContextUtils.getBean("dataBaseTableNameConfig");
            Map<String, String> tableNameMap = dataBaseTableNameConfig.getTableName();
            if (!CollectionUtils.isEmpty(tableNameMap)) {
                StatementHandler statementHandler = (StatementHandler) invocation.getTarget();
                MetaObject metaStatementHandler = MetaObject.forObject(statementHandler, new DefaultObjectFactory(),
                        new DefaultObjectWrapperFactory(), new DefaultReflectorFactory());
                BoundSql boundSql = (BoundSql) metaStatementHandler.getValue("delegate.boundSql");
    
                String sql = boundSql.getSql();
                List<SQLStatement> stmtList  = SQLUtils.parseStatements(sql, JdbcConstants.MYSQL);
                MySqlExportTableAliasVisitor visitor = new MySqlExportTableAliasVisitor();
                for (SQLStatement stmt : stmtList) {
                    stmt.accept(visitor);
                }
                String handleSQL = SQLUtils.toSQLString(stmtList, JdbcConstants.MYSQL);
    
                metaStatementHandler.setValue("delegate.boundSql.sql", handleSQL);
            }
            return invocation.proceed();
        }
    
        @Override
        public Object plugin(Object target) {
            return Plugin.wrap(target, this);
        }
    
        @Override
        public void setProperties(Properties properties) {
    
        }
    
    

    实际替换操作

    /**
     * @Auther: fatsnake
     * @Description":
     * @Date:2022/9/14 12:46
     * Copyright (c) 2022, zaodao All Rights Reserved.
     */
    public class MySqlExportTableAliasVisitor extends MySqlASTVisitorAdapter {
        @Override
        public boolean visit(SQLExprTableSource x) {
            DataBaseTableNameConfig dataBaseTableNameConfig
                    = (DataBaseTableNameConfig) SpringContextUtils.getBean("dataBaseTableNameConfig");
            Map<String, String> tableNameMap = dataBaseTableNameConfig.getTableName();
            if (tableNameMap.containsKey(x.getExpr().toString())) {
                x.setExpr(tableNameMap.get(x.getExpr().toString()));
            }
            return true;
        }
    } 
    

    三、尾声

    本来不了解这个mybatis的插件机制的,当时接到这个需求时,只是想着如何无感知的让业务开发人员使用,并且项目便于维护sql,最好不要维护两套sql,经过百度老师的指导,整合形成了这个结束解决方案。
    暴露出,对常用的中间件 还是不熟悉啊,源代码读的还是少……

    相关文章

      网友评论

          本文标题:根据环境不同,修改表名添加数据库名

          本文链接:https://www.haomeiwen.com/subject/feidartx.html