(1)MyBatis拦截器原理探究

(1)MyBatis拦截器原理探究,第1张

概要: 1介绍及配置、2源码分析、3 为何 拦截 这些方法    4注解 和 Plugin类使用

总结: 1)plugin()构建处理器(Handler)。interceptor()执行代理类。setProperties设置属性

           2)官网提供 @Interceptors和 @Signature及Plugin类,不一定直接用,可以抛弃,直接在plugin内部根据 target类型 *** 作

MyBatis提供 插件(plugin) ,插件是拦截器功能。允许插件拦截方法:

    Executor (update, query, flushStatements, commit, rollback, getTransaction, close, isClosed)

    ParameterHandler (getParameterObject, setParameters)

    ResultSetHandler (handleResultSets, handleOutputParameters)

    StatementHandler (prepare, parameterize, batch, update, query)

可拦截Executor接口部分方法如update,query,commit,rollback

拦截器拦截MyBatis中: 1拦截执行器方法、2拦截参数、3结果集、4Sql语法构建的处理

接口定义: 默认没有实现类,可实现符合自己需求拦截器,下面官网实例

全局xml配置

拦截Executor接口的update方法(就是SqlSession新增,删除,修改 *** 作)

1、 XMLConfigBuilder解析 MyBatis全局配置文件的 pluginElement私有方法 :通过 反射实例化plugin 节点中的 interceptor属性表示的类

2、 然后调 Configuration (全局配置类)的 addInterceptor 方法

3、interceptorChain 拦截器链  是 Configuration 内部属性,类型为InterceptorChain   定义:

理解拦截器 配置解析 及 归属 ,回过头看, 为何 拦截器会 拦截 这些方法(Executor,ParameterHandler,ResultSetHandler,StatementHandler的部分方法):

1) 4个方法 实例化对应对象 :都是 Configuration 方法。crud *** 作中都会 被执行到 ,顺序Executor,ParameterHandler,ResultSetHandler,StatementHandler

( ParameterHandler和ResultSetHandler的创建 是在 创建StatementHandler [3个可用的实现类CallableStatementHandler,PreparedStatementHandler,SimpleStatementHandler]时,其构造函数调用的[这3个实现类的构造函数其实都调用了父类BaseStatementHandler的构造函数])。

2)调 interceptorChain的 pluginAll 方法,InterceptorChain的pluginAll刚介绍过, 遍历所有拦截器 ,调各个拦截器plugin()。 注意:plugin返回值会直接赋值给原对象

3)由于可拦截StatementHandler ,主要处理sql,比如 分页 ,在拦截器plugin()中 处理 StatementHandler中的sql 即可 ,反射实现

1、@Intercepts和 @Signature

2、Plugin类的wrap方法:

实现InvocationHandler接口,返回了 JDK自身 提供 动态代理类

3、其中getSignatureMap方法:

1)先拿到拦截器这个类 @Interceptors注解 ,

2)拿注解属性 @ Signature 注解 集合 ,

3)遍历这个集合,拿@Signature的type属性(Class类型),

4)根据 type 得到有method和args属性的 Method 。

5)终会返回Map,type为key,value为Set<Method>,由于 @Interceptors的 @Signature属性是一个属性

如:@Interceptors返回 key为Executor,value为集合 (只有一个元素,Method实例,Executor的 update ,有MappedStatement和Object参数)。

Method实例是根据 @Signature的method和args属性得到的。如果args参数跟type类型的method方法对不上,抛异常

4、getAllInterfaces方法:

根据目标实例 target (可拦截的类,Executor,ParameterHandler,ResultSetHandler,StatementHandler)和它 父类们 ,返回 signatureMap 中含有 target 实现的 接口数组

Plugin类作用: 根据 @ Interceptors 注解,得注解属性 @ Signature 数组,根据每个 @Signature的 type,method,args 属性反射找到对应 Method 。根据调用 target 对象实现接口决定是否 返回代理 对象 替代 原先target对象。

如官网例子,当 Configuration 调 newExecutor 时,由于 Executor 的update(MappedStatement ms, Object parameter)被截获。返回代理类Plugin,不是Executor。如果是代理类,执行:Interceptor接口的 interceptor方法

proceed方法调用原先方法(不走代理)
>1foreach元素的属性有item, index, collection, open, separator, close;

item--->集合中每一个元素进行迭代的别名;

index--->表示迭代过程中,每次迭代的位置,

open--->该语句以什么开始;

separator--->在每次进行迭代之间以什么符号作为分隔符;

close--->以什么结束,

在使用foreach的时候最关键的也是容易出错的就是collection属性,

该属性是必须指定的,不同情况,属性值不同;

有一下3中情况:

① 如果传入的是单参数且参数类型是一个List的时候,collection属性值为list;

②如果传入的是单参数且参数类型是一个array数组的时候,collection的属性值为array;

③如果传入的参数是多个的时候,我们就需要把它们封装成一个Map了;

Dao层:

mybatis:

最近做个项目有很多的报表统计,涉及到几十张表抽取数据,形成一系列统计数据。
举个例子,比如统计一个单位不同职务级别女人数量、少数民族数量、30岁以上人员数量,40以上人员数量等。如:
董事长:0 0 0 1
中层领导:2 2 3 8
职员:20 40 80 200
刚开始想办法拼成了一个超级长的sql来做,但是最后长度超过vachar2的最大长度了,虽然有其他办法解决,虽然最大sql看起来很美好,但是想想后期的优化和维护,还是决定重新来规划。中间的过程就不说了,最终确定先定义一行数据存number型object(类似一维数组),形式如(0,0,1)。然后放到二维数组里。闲言少叙,如下定义:
[html] view plaincopy在CODE上查看代码片派生到我的代码片
create or replace
TYPE C_ROW IS OBJECT(
COUNT_SEX NUMBER, --性别女统计
COUNT_NATION NUMBER, --少数民族统计
COUNT_30AGE NUMBER --30岁统计
);
[html] view plaincopy在CODE上查看代码片派生到我的代码片
create or replace
TYPE C_ROW_CLM IS TABLE OF C_ROW;
关于以上的语法我就不解释了。C_ROW_CLM 就是二维数组。
一个简单的存储过程如下:就是往二维数组放了三列。

这个数组保存方法如下:
1、使用jpa,可以将数组存储为json格式,然后在实体类中使用@convert注解将json格式的数据转换为java对象。
2、使用mybatis,可以将数组存储为xml格式,然后在实体类中使用@xmlelement注解将xml格式的数据转换为java对象。
3、使用springdatapa,可以将数组存储为blob格式,然后在实体类中使用@lob注解将blob格式的数据转换为java对象。


欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/yw/13340325.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-07-17
下一篇 2023-07-17

发表评论

登录后才能评论

评论列表(0条)

保存