zoukankan      html  css  js  c++  java
  • 2021年12月14日复盘(Oracle Not In,Limit 1000)

    1、遇到Oracle Not In 无效的问题,原因是Not In里面的子查询结果有空值,需要过滤掉

    2、Oracle Limit 1000的问题,自己按照Mybatis Plus的租户拦截器做了修改

      1)、重点需要理解下表达树,这个刚好旁边大佬学历高,跟我普及了下二叉树用来做数学公式计算的原理

      2)、需要写递归,拆分左右节点类型的,比如Or或And,然后也要拆包括号表达式,最终处理In表达式,其他类型的直接返回

      3)、版本是3.4.2

    import com.baomidou.mybatisplus.core.parser.SqlParserHelper;
    import com.baomidou.mybatisplus.core.plugins.InterceptorIgnoreHelper;
    import com.baomidou.mybatisplus.core.toolkit.CollectionUtils;
    import com.baomidou.mybatisplus.core.toolkit.PluginUtils;
    import com.baomidou.mybatisplus.core.toolkit.StringPool;
    import com.baomidou.mybatisplus.extension.parser.JsqlParserSupport;
    import com.baomidou.mybatisplus.extension.plugins.inner.InnerInterceptor;
    import lombok.Data;
    import lombok.EqualsAndHashCode;
    import lombok.NoArgsConstructor;
    import lombok.ToString;
    import net.sf.jsqlparser.expression.BinaryExpression;
    import net.sf.jsqlparser.expression.Expression;
    import net.sf.jsqlparser.expression.NotExpression;
    import net.sf.jsqlparser.expression.Parenthesis;
    import net.sf.jsqlparser.expression.operators.conditional.OrExpression;
    import net.sf.jsqlparser.expression.operators.relational.ExistsExpression;
    import net.sf.jsqlparser.expression.operators.relational.ExpressionList;
    import net.sf.jsqlparser.expression.operators.relational.InExpression;
    import net.sf.jsqlparser.expression.operators.relational.ItemsList;
    import net.sf.jsqlparser.schema.Table;
    import net.sf.jsqlparser.statement.delete.Delete;
    import net.sf.jsqlparser.statement.insert.Insert;
    import net.sf.jsqlparser.statement.select.*;
    import net.sf.jsqlparser.statement.update.Update;
    import org.apache.commons.lang3.StringUtils;
    import org.apache.ibatis.executor.Executor;
    import org.apache.ibatis.executor.statement.StatementHandler;
    import org.apache.ibatis.mapping.BoundSql;
    import org.apache.ibatis.mapping.MappedStatement;
    import org.apache.ibatis.mapping.SqlCommandType;
    import org.apache.ibatis.session.ResultHandler;
    import org.apache.ibatis.session.RowBounds;
    
    import java.sql.Connection;
    import java.util.List;
    import java.util.stream.Collectors;
    
    /**
     * @author linjiabin
     * @since 1.0.0
     */
    @Data
    @NoArgsConstructor
    @ToString(callSuper = true)
    @EqualsAndHashCode(callSuper = true)
    public class OracleLimit1000InnerInterceptor extends JsqlParserSupport implements InnerInterceptor {
    
    
        @Override
        public void beforeQuery(Executor executor, MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) {
            if (InterceptorIgnoreHelper.willIgnoreTenantLine(ms.getId())) return;
            if (SqlParserHelper.getSqlParserInfo(ms)) return;
            PluginUtils.MPBoundSql mpBs = PluginUtils.mpBoundSql(boundSql);
            try {
                String sql = mpBs.sql();
                if (needRebuild(sql)) {
                    mpBs.sql(parserSingle(sql, null));
                }
            } catch (Exception e) {
                logger.error(e.getMessage(), e);
            }
        }
    
        private boolean needRebuild(String sql) {
            return StringUtils.isNotBlank(sql) && sql.toUpperCase().contains(" IN ");
        }
    
        @Override
        public void beforePrepare(StatementHandler sh, Connection connection, Integer transactionTimeout) {
            PluginUtils.MPStatementHandler mpSh = PluginUtils.mpStatementHandler(sh);
            MappedStatement ms = mpSh.mappedStatement();
            SqlCommandType sct = ms.getSqlCommandType();
            if (sct == SqlCommandType.UPDATE || sct == SqlCommandType.DELETE) {
                if (InterceptorIgnoreHelper.willIgnoreTenantLine(ms.getId())) return;
                if (SqlParserHelper.getSqlParserInfo(ms)) return;
                PluginUtils.MPBoundSql mpBs = mpSh.mPBoundSql();
                try {
                    String sql = mpBs.sql();
                    if (needRebuild(sql)) {
                        String parserMulti = parserMulti(sql, null);
                        if (sql.endsWith(StringPool.SEMICOLON) && !parserMulti.endsWith(StringPool.SEMICOLON)) {
                            parserMulti += StringPool.SEMICOLON;
                        }
                        mpBs.sql(parserMulti);
                    }
                } catch (Exception e) {
                    logger.error(e.getMessage(), e);
                }
            }
        }
    
        @Override
        protected void processSelect(Select select, int index, String sql, Object obj) {
            processSelectBody(select.getSelectBody());
            List<WithItem> withItemsList = select.getWithItemsList();
            if (!CollectionUtils.isEmpty(withItemsList)) {
                withItemsList.forEach(this::processSelectBody);
            }
        }
    
        protected void processSelectBody(SelectBody selectBody) {
            if (selectBody == null) {
                return;
            }
            if (selectBody instanceof PlainSelect) {
                processPlainSelect((PlainSelect) selectBody);
            } else if (selectBody instanceof WithItem) {
                WithItem withItem = (WithItem) selectBody;
                processSelectBody(withItem.getSelectBody());
            } else {
                SetOperationList operationList = (SetOperationList) selectBody;
                if (operationList.getSelects() != null && !operationList.getSelects().isEmpty()) {
                    operationList.getSelects().forEach(this::processSelectBody);
                }
            }
        }
    
        @Override
        protected void processInsert(Insert insert, int index, String sql, Object obj) {
            // no do anything at insert
        }
    
        /**
         * update 语句处理
         */
        @Override
        protected void processUpdate(Update update, int index, String sql, Object obj) {
            update.setWhere(this.andExpression(update.getWhere()));
        }
    
        /**
         * delete 语句处理
         */
        @Override
        protected void processDelete(Delete delete, int index, String sql, Object obj) {
            delete.setWhere(this.andExpression(delete.getWhere()));
        }
    
        /**
         * delete update select 语句 where 处理
         */
        protected Expression andExpression(Expression where) {
            // 遇到左右表达式类型的,继续递归
            if (where instanceof BinaryExpression) {
                BinaryExpression binaryExpression = (BinaryExpression) where;
                Expression rightExpression = binaryExpression.getRightExpression();
                binaryExpression.setRightExpression(andExpression(rightExpression));
                Expression leftExpression = binaryExpression.getLeftExpression();
                binaryExpression.setLeftExpression(andExpression(leftExpression));
            }
            // 遇到括号类型的,拆包递归
            if (where instanceof Parenthesis) {
                Parenthesis parenthesis = (Parenthesis) where;
                return new Parenthesis(andExpression(parenthesis.getExpression()));
            }
            // 遇到in表达式的时候,尝试拆分
            if (where instanceof InExpression) {
                return builderExpression((InExpression) where);
            }
            // 其他表达式直接返回
            return where;
        }
    
        /**
         * 处理 PlainSelect
         */
        protected void processPlainSelect(PlainSelect plainSelect) {
            FromItem fromItem = plainSelect.getFromItem();
            Expression where = plainSelect.getWhere();
            processWhereSubSelect(where);
            if (fromItem instanceof Table) {
                plainSelect.setWhere(builderExpression(where));
            } else {
                processFromItem(fromItem);
            }
            List<Join> joins = plainSelect.getJoins();
            if (joins != null && !joins.isEmpty()) {
                joins.forEach(j -> {
                    processJoin(j);
                    processFromItem(j.getRightItem());
                });
            }
        }
    
        /**
         * 处理where条件内的子查询
         * <p>
         * 支持如下:
         * 1. in
         * 2. =
         * 3. >
         * 4. <
         * 5. >=
         * 6. <=
         * 7. <>
         * 8. EXISTS
         * 9. NOT EXISTS
         * <p>
         * 前提条件:
         * 1. 子查询必须放在小括号中
         * 2. 子查询一般放在比较操作符的右边
         *
         * @param where where 条件
         */
        protected void processWhereSubSelect(Expression where) {
            if (where == null) {
                return;
            }
            if (where instanceof FromItem) {
                processFromItem((FromItem) where);
                return;
            }
            if (where.toString().contains("SELECT")) {
                // 有子查询
                if (where instanceof BinaryExpression) {
                    // 比较符号 , and , or , 等等
                    BinaryExpression expression = (BinaryExpression) where;
                    processWhereSubSelect(expression.getLeftExpression());
                    processWhereSubSelect(expression.getRightExpression());
                } else if (where instanceof InExpression) {
                    // in
                    InExpression expression = (InExpression) where;
                    ItemsList itemsList = expression.getRightItemsList();
                    if (itemsList instanceof SubSelect) {
                        processSelectBody(((SubSelect) itemsList).getSelectBody());
                    }
                } else if (where instanceof ExistsExpression) {
                    // exists
                    ExistsExpression expression = (ExistsExpression) where;
                    processWhereSubSelect(expression.getRightExpression());
                } else if (where instanceof NotExpression) {
                    // not exists
                    NotExpression expression = (NotExpression) where;
                    processWhereSubSelect(expression.getExpression());
                } else if (where instanceof Parenthesis) {
                    Parenthesis expression = (Parenthesis) where;
                    processWhereSubSelect(expression.getExpression());
                }
            }
        }
    
        /**
         * 处理子查询等
         */
        protected void processFromItem(FromItem fromItem) {
            if (fromItem instanceof SubJoin) {
                SubJoin subJoin = (SubJoin) fromItem;
                if (subJoin.getJoinList() != null) {
                    subJoin.getJoinList().forEach(this::processJoin);
                }
                if (subJoin.getLeft() != null) {
                    processFromItem(subJoin.getLeft());
                }
            } else if (fromItem instanceof SubSelect) {
                SubSelect subSelect = (SubSelect) fromItem;
                if (subSelect.getSelectBody() != null) {
                    processSelectBody(subSelect.getSelectBody());
                }
            } else if (fromItem instanceof ValuesList) {
                logger.debug("Perform a subquery, if you do not give us feedback");
            } else if (fromItem instanceof LateralSubSelect) {
                LateralSubSelect lateralSubSelect = (LateralSubSelect) fromItem;
                if (lateralSubSelect.getSubSelect() != null) {
                    SubSelect subSelect = lateralSubSelect.getSubSelect();
                    if (subSelect.getSelectBody() != null) {
                        processSelectBody(subSelect.getSelectBody());
                    }
                }
            }
        }
    
        /**
         * 处理联接语句
         */
        protected void processJoin(Join join) {
            if (join.getRightItem() instanceof Table) {
                join.setOnExpression(builderExpression(join.getOnExpression()));
            }
        }
    
        /**
         * 处理条件
         */
        protected Expression builderExpression(Expression currentExpression) {
            if (currentExpression == null) {
                return null;
            }
            return andExpression(currentExpression);
        }
    
        protected Expression builderExpression(InExpression inExpression) {
            Expression leftExpression = inExpression.getLeftExpression();
            ItemsList rightItemsList = inExpression.getRightItemsList();
            if (rightItemsList instanceof ExpressionList) {
    
                ExpressionList expressionList = (ExpressionList) rightItemsList;
                List<Expression> expressions = expressionList.getExpressions();
                int size = expressions.size();
                int limit = 1000;
                if (size > limit) {
                    OrExpression root = new OrExpression();
                    int step = size / limit + 1;
                    root.setLeftExpression(new InExpression(leftExpression, new ExpressionList(expressions.subList(0, limit))));
                    if (step == 2) {
                        int toIndex = getToIndex(size, limit);
                        root.setRightExpression(new InExpression(leftExpression, new ExpressionList(expressions.subList(limit, toIndex))));
                        return root;
                    }
                    OrExpression orExpression = new OrExpression();
                    root.setRightExpression(orExpression);
                    for (int i = 1; i < step; i++) {
                        List<Expression> segment = expressions.stream().skip((long) i * limit)
                                .limit(limit).collect(Collectors.toList());
                        if (i == step - 2) {
                            orExpression.setLeftExpression(new InExpression(leftExpression, new ExpressionList(segment)));
                            List<Expression> last = expressions.stream().skip((long) (i + 1) * limit).collect(Collectors.toList());
                            orExpression.setRightExpression(new InExpression(leftExpression, new ExpressionList(last)));
                            break;
                        } else {
                            OrExpression orExpression1 = new OrExpression();
                            orExpression.setLeftExpression(new InExpression(leftExpression, new ExpressionList(segment)));
                            orExpression.setRightExpression(orExpression1);
                            orExpression = orExpression1;
                        }
                    }
                    return new Parenthesis(root);
                }
            }
            return inExpression;
        }
    
        private int getToIndex(int size, int limit) {
            int toIndex = limit * 2;
            if (toIndex > size) {
                toIndex = size;
            }
            return toIndex;
        }
    
    
    }

    由于遇到了update 自动去除最后一个分号的问题,追加了一个判断和处理

    并且将重组sql的方法用异常捕获包起来,避免因为重组sql导致出错而无法继续的情况

    简单地过滤掉不需要解析的场景,目前仅判断是否包含in关键词

  • 相关阅读:
    Scott Mitchell 的ASP.NET 2.0数据教程
    asp.net单点登录 一个账号只能在一个地方登录
    javascript关键字加亮加连接
    Windows环境变量
    IE支持HTML5的解决方法
    JavaScript test() 方法及判断是否为iPhone|iPad|iPod
    《C#4.0中文视频教程共20课完整版》下载
    DataView.RowFilter 的使用
    SQL Server中如何备份到异机
    完美兼容IE、FF、Opera的Ajax类支持get、post、自定义回调函数
  • 原文地址:https://www.cnblogs.com/gabin/p/15689744.html
Copyright © 2011-2022 走看看