总结: 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方法调用原先方法(不走代理)
>
答案:
<if test="searchdataforArray != null">
and datefor in ;
<foreach item="item" index="index";
collection="searchdataforArray";
open="("separator="," close=")">#{item}</foreach>
</if>。
循环体中的具体对象:支持属性的点路径访问,如itemage,iteminfodetails,具体说明:在list和数组中是其中的对象,在map中是value,collection,要做foreach的对象,作为入参时,List<>对象默认用list代替作为键,数组对象有array代替作为键,Map对象没有默认的键。当然在作为入参时可以使用@Param("keyName")来设置键,设置keyName后,list,array将会失效。
除了入参这种情况外,还有一种作为参数对象的某个字段的时候。
举个例子,比如统计一个单位不同职务级别女人数量、少数民族数量、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 就是二维数组。
一个简单的存储过程如下:就是往二维数组放了三列。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)