获取MapperedStatement

  1. MappedStatement mappedStatement = (MappedStatement) args[0];

判断sql类型

  1. if (!SqlCommandType.SELECT.equals(mappedStatement.getSqlCommandType())) {
  2. return invocation.proceed();
  3. }

获取拦截的方法

  1. // 获取执行方法的包名和方法名
  2. String namespace = mappedStatement.getId();
  3. String className = namespace.substring(0, namespace.lastIndexOf("."));
  4. String mapperName = className.substring(className.lastIndexOf(".") + 1, className.length());
  5. String methodName = namespace.substring(namespace.lastIndexOf(".") + 1, namespace.length());

获取参数

  1. Object parameter = args[1];

获取BoundSql

  1. BoundSql boundSql = mappedStatement.getBoundSql(parameter);
  2. String originalSql = boundSql.getSql();
  3. Object parameterObject = boundSql.getParameterObject();

重置mappedStatement

  1. /**
  2. * 包装sql后,重置到invocation中
  3. *
  4. * @param invocation
  5. * @param sql
  6. * @throws SQLException
  7. */
  8. private void resetSql2Invocation(Invocation invocation, String sql) throws SQLException {
  9. final Object[] args = invocation.getArgs();
  10. MappedStatement statement = (MappedStatement) args[0];
  11. Object parameterObject = args[1];
  12. BoundSql boundSql = statement.getBoundSql(parameterObject);
  13. MappedStatement newStatement = newMappedStatement(statement, new BoundSqlSqlSource(boundSql));
  14. MetaObject msObject = MetaObject.forObject(newStatement, new DefaultObjectFactory(), new DefaultObjectWrapperFactory(), new DefaultReflectorFactory());
  15. msObject.setValue("sqlSource.boundSql.sql", sql);
  16. args[0] = newStatement;
  17. }
  18. private MappedStatement newMappedStatement(MappedStatement ms, SqlSource newSqlSource) {
  19. MappedStatement.Builder builder =
  20. new MappedStatement.Builder(ms.getConfiguration(), ms.getId(), newSqlSource, ms.getSqlCommandType());
  21. builder.resource(ms.getResource());
  22. builder.fetchSize(ms.getFetchSize());
  23. builder.statementType(ms.getStatementType());
  24. builder.keyGenerator(ms.getKeyGenerator());
  25. if (ms.getKeyProperties() != null && ms.getKeyProperties().length != 0) {
  26. StringBuilder keyProperties = new StringBuilder();
  27. for (String keyProperty : ms.getKeyProperties()) {
  28. keyProperties.append(keyProperty).append(",");
  29. }
  30. keyProperties.delete(keyProperties.length() - 1, keyProperties.length());
  31. builder.keyProperty(keyProperties.toString());
  32. }
  33. builder.timeout(ms.getTimeout());
  34. builder.parameterMap(ms.getParameterMap());
  35. builder.resultMaps(ms.getResultMaps());
  36. builder.resultSetType(ms.getResultSetType());
  37. builder.cache(ms.getCache());
  38. builder.flushCacheRequired(ms.isFlushCacheRequired());
  39. builder.useCache(ms.isUseCache());
  40. return builder.build();
  41. }
  42. // 定义一个内部辅助类,作用是包装sql
  43. class BoundSqlSqlSource implements SqlSource {
  44. private BoundSql boundSql;
  45. public BoundSqlSqlSource(BoundSql boundSql) {
  46. this.boundSql = boundSql;
  47. }
  48. @Override
  49. public BoundSql getBoundSql(Object parameterObject) {
  50. return boundSql;
  51. }
  52. }