MyBatis攔截器的實現原理
前言
Mybatis攔截器並不是每個對象裡面的方法都可以被攔截的。Mybatis攔截器隻能攔截Executor、StatementHandler、ParameterHandler、ResultSetHandler四個類裡面的方法,這四個對象在創建的時候才會創建代理。
用途:實際工作中,可以使用Mybatis攔截器來做一些SQL權限校驗、數據過濾、數據加密脫敏、SQL執行時間性能監控和告警等。
1.使用方法
以在Spring中創建 StatementHandler.update()方法的攔截器為例:
@Component @Order(1) @Intercepts({@Signature(type = StatementHandler.class, method = "update", args = {Statement.class}),}) public class SqlValidateMybatisInterceptor extends PRSMybatisInterceptor { @Override protected Object before(Invocation invocation) throws Throwable { String sql=""; Statement statement=(Statement) invocation.getArgs()[0]; if(Proxy.isProxyClass(statement.getClass())){ MetaObject metaObject= SystemMetaObject.forObject(statement); Object h=metaObject.getValue("h"); if(h instanceof StatementLogger){ RoutingStatementHandler rsh=(RoutingStatementHandler) invocation.getTarget(); sql=rsh.getBoundSql().getSql(); }else { PreparedStatementLogger psl=(PreparedStatementLogger) h; sql=psl.getPreparedStatement().toString(); } }else{ sql=statement.toString(); } if(containsDelete(sql)&&!containsWhere(sql)){ throw new SQLException("不能刪除整張表,sql:"+sql); } return null; } private boolean containsDelete(String sql){ return sql.contains("delete")||sql.contains("DELETE"); } private boolean containsWhere(String sql){ return sql.contains("where")||sql.contains("WHERE"); } } public class PRSMybatisInterceptor implements Interceptor { Boolean needBreak=false; @Override public Object intercept(Invocation invocation) throws Throwable { Object result= before(invocation); if(needBreak){ return result; } result= invocation.proceed(); result=after(result,invocation); return result; } protected Object before(Invocation invocation) throws Throwable{ return null; } protected Object after(Object result,Invocation invocation) throws Throwable{ return result; } @Override public Object plugin(Object o) { return Plugin.wrap(o, this); } @Override public void setProperties(Properties properties) { } }
1. 自定義攔截器 實現 org.apache.ibatis.plugin.Interceptor 接口與其中的方法。在plugin方法中需要返回 return Plugin.wrap(o, this)。在intercept方法中可以實現攔截的業務邏輯,改方法的 參數 Invocation中有原始調用的 對象,方法和參數,可以對其任意處理。
2. 在自定義的攔截器上添加需要攔截的對象和方法,通過註解 org.apache.ibatis.plugin.Intercepts 添加。如示例代碼所示:
Intercepts的值是一個簽名數組,簽名中包含要攔截的 類,方法和參數。
2.MyBatis對象的創建
代理對象指的是:可以被攔截的4個類的實例。
代理對象創建時需要解析攔截器,從而利用JDK動態代理將攔截器的邏輯織入原始對象。
DefaultSqlSession中依賴Executor,如果新建的時候會創建executor
private SqlSession openSessionFromConnection(ExecutorType execType, Connection connection) { ... final Executor executor = configuration.newExecutor(tx, execType); return new DefaultSqlSession(configuration, executor, autoCommit); } public Executor newExecutor(Transaction transaction, ExecutorType executorType) { executorType = executorType == null ? defaultExecutorType : executorType; executorType = executorType == null ? ExecutorType.SIMPLE : executorType; Executor executor; if (ExecutorType.BATCH == executorType) { executor = new BatchExecutor(this, transaction); } else if (ExecutorType.REUSE == executorType) { executor = new ReuseExecutor(this, transaction); } else { executor = new SimpleExecutor(this, transaction); } if (cacheEnabled) { executor = new CachingExecutor(executor); } executor = (Executor) interceptorChain.pluginAll(executor); return executor; }
Executor中要用StatementHandler執行sql語句,StatementHandler是調用configuration.newStatementHandler()方法創建的。
StatementHandler handler = configuration.newStatementHandler(wrapper, ms, parameterObject, rowBounds, resultHandler, boundSql); public StatementHandler newStatementHandler(Executor executor, MappedStatement mappedStatement, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) { StatementHandler statementHandler = new RoutingStatementHandler(executor, mappedStatement, parameterObject, rowBounds, resultHandler, boundSql); statementHandler = (StatementHandler) interceptorChain.pluginAll(statementHandler); return statementHandler; }
StatementHandler依賴 parameterHandler 和 resultSetHandler,在構造 StatementHandler 時會調用一下方法創建這兩個 handler。
this.parameterHandler = configuration.newParameterHandler(mappedStatement, parameterObject, boundSql); public ParameterHandler newParameterHandler(MappedStatement mappedStatement, Object parameterObject, BoundSql boundSql) { ParameterHandler parameterHandler = mappedStatement.getLang().createParameterHandler(mappedStatement, parameterObject, boundSql); parameterHandler = (ParameterHandler) interceptorChain.pluginAll(parameterHandler); return parameterHandler; }
this.resultSetHandler = configuration.newResultSetHandler(executor, mappedStatement, rowBounds, parameterHandler, resultHandler, boundSql); public ResultSetHandler newResultSetHandler(Executor executor, MappedStatement mappedStatement, RowBounds rowBounds, ParameterHandler parameterHandler, ResultHandler resultHandler, BoundSql boundSql) { ResultSetHandler resultSetHandler = new DefaultResultSetHandler(executor, mappedStatement, parameterHandler, resultHandler, boundSql, rowBounds); resultSetHandler = (ResultSetHandler) interceptorChain.pluginAll(resultSetHandler); return resultSetHandler; }
3.代理對象的創建
3.1 攔截器的獲取
從對象的創建過程中可以看出 代理 對象的創建時通過 InterceptorChain.pluginAll() 方法創建的。
查看 攔截器鏈 InterceptorChain 發現,其中的攔截器的添加是在 Configuration 中。因為攔截器被聲明為Bean瞭,所以在MyBatis初始化的時候,會掃描所有攔截器,添加到 InterceptorChain 中。
3.2 代理對象的創建
從上一步得知代理對象的創建是調用 Interceptor.pugin() 方法,然後調用 Plugin.wrap() 方法
Interceptor @Override public Object plugin(Object o) { return Plugin.wrap(o, this); }
Plugin實現瞭 InvocationHandler 接口
在 Plugin.wrap() 方法中會獲取當前攔截器的接口,生成動態代理。
4. 攔截器的執行過程
在動態代理中當代理對象調用方法時,會將方法的調用委托給 InvocationHandler,也就是 Plugin,
如下圖所示;
在該方法中 獲取攔截器簽名中的方法,如果包含當前方法,則調用攔截方法,否則執行原方法的調用。
5. 攔截器的執行順序
攔截器的順序配置使用 Spring 中的 org.springframework.core.annotation.Order 註解配置。
order值大的攔截器先執行,order值大的在interceptors中越靠後,最後生成代理,所以先執行。
到此這篇關於MyBatis攔截器的實現原理的文章就介紹到這瞭,更多相關MyBatis攔截器 內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- Java mybatis 開發自定義插件
- 詳解MyBatis工作原理
- 在springboot中如何給mybatis加攔截器
- mybatis源碼解讀之executor包語句處理功能
- Mybatis第三方PageHelper分頁插件的使用與原理