前言
站在用戶的角度思考問題,與客戶深入溝通,找到榮縣網(wǎng)站設計與榮縣網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗,讓設計與互聯(lián)網(wǎng)技術結(jié)合,創(chuàng)造個性化、用戶體驗好的作品,建站類型包括:做網(wǎng)站、成都做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機端網(wǎng)站、網(wǎng)站推廣、域名與空間、雅安服務器托管、企業(yè)郵箱。業(yè)務覆蓋榮縣地區(qū)。
Mybatis作為一個應用廣泛的優(yōu)秀的ORM框架,已經(jīng)成了JavaWeb世界近乎標配的部分,這個框架具有強大的靈活性,在四大組件(Executor、StatementHandler、ParameterHandler、ResultSetHandler)處提供了簡單易用的插件擴展機制。Mybatis對持久層的操作就是借助于四大核心對象。MyBatis支持用插件對四大核心對象進行攔截,對mybatis來說插件就是攔截器,用來增強核心對象的功能,增強功能本質(zhì)上是借助于底層的動態(tài)代理實現(xiàn)的,換句話說,MyBatis中的四大對象都是代理對象。
四大核心對象簡介
MyBatis 四大核心對象
ParameterHandler:處理SQL的參數(shù)對象
ResultSetHandler:處理SQL的返回結(jié)果集
StatementHandler:數(shù)據(jù)庫的處理對象,用于執(zhí)行SQL語句
Executor:MyBatis的執(zhí)行器,用于執(zhí)行增刪改查操作
Mybatis插件原理
攔截
插件具體是如何攔截并附加額外的功能的呢?
以ParameterHandler 來說
public ParameterHandler newParameterHandler(MappedStatement mappedStatement, Object object, BoundSql sql, InterceptorChain interceptorChain){ ParameterHandler parameterHandler = mappedStatement.getLang().createParameterHandler(mappedStatement,object,sql); parameterHandler = (ParameterHandler) interceptorChain.pluginAll(parameterHandler); return parameterHandler; } public Object pluginAll(Object target) { for (Interceptor interceptor : interceptors) { target = interceptor.plugin(target); } return target; }
interceptorChain 保存了所有的攔截器(interceptors),是mybatis初始化的時候創(chuàng)建的。調(diào)用攔截器鏈中的攔截器依次的對目標進行攔截或增強。interceptor.plugin(target)中的target就可以理解為mybatis中的四大對象。返回的target是被重重代理后的對象。
插件接口
Mybatis插件接口-Interceptor
1.Intercept方法,插件的核心方法
2.plugin方法,生成target的代理對象
3.setProperties方法,傳遞插件所需參數(shù)
插件實例
插件開發(fā)需要以下步驟
/** 插件簽名,告訴mybatis單錢插件用來攔截那個對象的哪個方法 **/ @Intercepts({@Signature(type = ResultSetHandler.class,method ="handleResultSets",args = Statement.class)}) public class MyFirstInterceptor implements Interceptor { /** @Description 攔截目標對象的目標方法 **/ @Override public Object intercept(Invocation invocation) throws Throwable { System.out.println("攔截的目標對象:"+invocation.getTarget()); Object object = invocation.proceed(); return object; } /** * @Description 包裝目標對象 為目標對象創(chuàng)建代理對象 * @Param target為要攔截的對象 * @Return 代理對象 */ @Override public Object plugin(Object target) { System.out.println("將要包裝的目標對象:"+target); return Plugin.wrap(target,this); } /** 獲取配置文件的屬性 **/ @Override public void setProperties(Properties properties) { System.out.println("插件配置的初始化參數(shù):"+properties); } }
在mybatis.xml中配置插件
調(diào)用查詢方法,查詢方法會返回ResultSet
public class MyBatisTest { public static SqlSessionFactory sqlSessionFactory = null; public static SqlSessionFactory getSqlSessionFactory() { if (sqlSessionFactory == null) { String resource = "mybatis-config.xml"; try { Reader reader = Resources.getResourceAsReader(resource); sqlSessionFactory = new SqlSessionFactoryBuilder().build(reader); } catch (IOException e) { e.printStackTrace(); } } return sqlSessionFactory; } public void testGetById() { SqlSession sqlSession = this.getSqlSessionFactory().openSession(); PersonMapper personMapper = sqlSession.getMapper(PersonMapper.class); Person person=personMapper.getById(2001); System.out.println(person.toString()); } public static void main(String[] args) { new MyBatisTest().testGetById(); } }
輸出結(jié)果
插件配置的初始化參數(shù):{name=Bob}
將要包裝的目標對象:org.apache.ibatis.executor.CachingExecutor@754ba872
將要包裝的目標對象:org.apache.ibatis.scripting.defaults.DefaultParameterHandler@192b07fd
將要包裝的目標對象:org.apache.ibatis.executor.resultset.DefaultResultSetHandler@7e0b0338
將要包裝的目標對象:org.apache.ibatis.executor.statement.RoutingStatementHandler@1e127982
攔截的目標對象:org.apache.ibatis.executor.resultset.DefaultResultSetHandler@7e0b0338
Person{id=2001, username='Tom', email='email@0', gender='F'}
多插件開發(fā)過程
1.創(chuàng)建代理對象時,按照插件配置的順序進行包裝
2.執(zhí)行目標方法后,是按照代理的逆向進行執(zhí)行
總結(jié)
1.遵循插件盡量不使用的原則,因為會修改底層設計
2.插件是生成的層層代理對象的責任鏈模式,使用反射機制實現(xiàn)
3.插件的編寫要考慮全面,特別是多個插件層層代理的時候
好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對創(chuàng)新互聯(lián)的支持。