mybatis插件机制的示例分析
这篇文章主要为大家展示了"mybatis插件机制的示例分析",内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下"mybatis插件机制的示例分析"这篇文章吧。
前言
Mybatis作为一个应用广泛的优秀的ORM框架,已经成了JavaWeb世界近乎标配的部分,这个框架具有强大的灵活性,在四大组件(Executor、StatementHandler、ParameterHandler、ResultSetHandler)处提供了简单易用的插件扩展机制。Mybatis对持久层的操作就是借助于四大核心对象。MyBatis支持用插件对四大核心对象进行拦截,对mybatis来说插件就是拦截器,用来增强核心对象的功能,增强功能本质上是借助于底层的动态代理实现的,换句话说,MyBatis中的四大对象都是代理对象。
四大核心对象简介
MyBatis 四大核心对象
ParameterHandler:处理SQL的参数对象
ResultSetHandler:处理SQL的返回结果集
StatementHandler:数据库的处理对象,用于执行SQL语句
Executor:MyBatis的执行器,用于执行增删改查操作
Mybatis插件原理
Mybatis的插件借助于责任链的模式进行对拦截的处理 使用动态代理对目标对象进行包装,达到拦截的目的 作用于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初始化的时候创建的。调用拦截器链中的拦截器依次的对目标进行拦截或增强。interceptor.plugin(target)中的target就可以理解为mybatis中的四大对象。返回的target是被重重代理后的对象。
插件接口
Mybatis插件接口-Interceptor
1.Intercept方法,插件的核心方法
2.plugin方法,生成target的代理对象
3.setProperties方法,传递插件所需参数
插件实例
插件开发需要以下步骤
自定义插件需要实现上述接口 增加@Intercepts注解(声明是哪个核心组件的插件,以及对哪些方法进行扩展) 在xml文件中配置插件
/** 插件签名,告诉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 包装目标对象 为目标对象创建代理对象 * @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("插件配置的初始化参数:"+properties); }}
在mybatis.xml中配置插件
调用查询方法,查询方法会返回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(); }}
输出结果
插件配置的初始化参数:{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@7e0b0338Person{id=2001, username='Tom', email='email@0', gender='F'}
多插件开发过程
1.创建代理对象时,按照插件配置的顺序进行包装
2.执行目标方法后,是按照代理的逆向进行执行
以上是"mybatis插件机制的示例分析"这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注行业资讯频道!