mybatis如何实现SQL查询拦截修改
这篇文章将为大家详细讲解有关mybatis如何实现SQL查询拦截修改,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。
前言
截器的一个作用就是我们可以拦截某些方法的调用,我们可以选择在这些被拦截的方法执行前后加上某些逻辑,也可以在执行这些被拦截的方法时执行自己的逻辑而不再执行被拦截的方法。
Mybatis拦截器设计的一个初衷就是为了供用户在某些时候可以实现自己的逻辑而不必去动Mybatis固有的逻辑。比如我想针对所有的SQL执行某个固定的操作,针对SQL查询执行安全检查,或者记录相关SQL查询日志等等。
Mybatis为我们提供了一个Interceptor接口,可以实现自定义的拦截器。
publicinterfaceInterceptor{Objectintercept(Invocationinvocation)throwsThrowable;Objectplugin(Objecttarget);voidsetProperties(Propertiesproperties);}
接口中包含了三个方法定义
intercept方法为具体的拦截对象的处理方法,传入的Invocation包含了拦截目标类的实力,拦截的方法和方法的入参数组。使用Invocation的procced执行原函数。
plugin 中执行判断是否要进行拦截进,如果不需要拦截,直接返回target,如果需要拦截则调用Plugin类中的wrap静态方法,如果当前拦截器实现了任意接口,则返回一个代理对象,否则直接返回(回忆代理模式的设计)。代理对象实际是一个Plugin类实例,它实现了InvocationHandler接口 ,InvocationHandler接口仅包含invoke方法用于回调方法。
当执行代理对象的接口方法时,会调用Plugin的invoke方法,它会把要执行的对象,方法和参数打包成Invocation对象传给拦截器的intercept方法。Invocation定义了一个procced方法,用于执行被拦截的原方法。
Plugin类定义
publicclassPluginimplementsInvocationHandler{privateObjecttarget;privateInterceptorinterceptor;privateMap,Set>signatureMap;privatePlugin(Objecttarget,Interceptorinterceptor,Map,Set>signatureMap){this.target=target;this.interceptor=interceptor;this.signatureMap=signatureMap;}publicstaticObjectwrap(Objecttarget,Interceptorinterceptor){Map,Set>signatureMap=getSignatureMap(interceptor);Classtype=target.getClass();Class[]interfaces=getAllInterfaces(type,signatureMap);if(interfaces.length>0){returnProxy.newProxyInstance(type.getClassLoader(),interfaces,newPlugin(target,interceptor,signatureMap));}returntarget;}publicObjectinvoke(Objectproxy,Methodmethod,Object[]args)throwsThrowable{try{Setmethods=signatureMap.get(method.getDeclaringClass());if(methods!=null&&methods.contains(method)){returninterceptor.intercept(newInvocation(target,method,args));}returnmethod.invoke(target,args);}catch(Exceptione){throwExceptionUtil.unwrapThrowable(e);}}privatestaticMap,Set>getSignatureMap(Interceptorinterceptor){InterceptsinterceptsAnnotation=interceptor.getClass().getAnnotation(Intercepts.class);if(interceptsAnnotation==null){//issue#251thrownewPluginException("No@Interceptsannotationwasfoundininterceptor"+interceptor.getClass().getName());}Signature[]sigs=interceptsAnnotation.value();Map,Set>signatureMap=newHashMap,Set>();for(Signaturesig:sigs){Setmethods=signatureMap.get(sig.type());if(methods==null){methods=newHashSet();signatureMap.put(sig.type(),methods);}try{Methodmethod=sig.type().getMethod(sig.method(),sig.args());methods.add(method);}catch(NoSuchMethodExceptione){thrownewPluginException("Couldnotfindmethodon"+sig.type()+"named"+sig.method()+".Cause:"+e,e);}}returnsignatureMap;}privatestaticClass[]getAllInterfaces(Classtype,Map,Set>signatureMap){Set>interfaces=newHashSet>();while(type!=null){for(Classc:type.getInterfaces()){if(signatureMap.containsKey(c)){interfaces.add(c);}}type=type.getSuperclass();}returninterfaces.toArray(newClass[interfaces.size()]);}}
setProperties 方法顾名思义,用于设置属性的。bean的属性初始化方法有很多,这是其中的一种。
mybatis提供了@Intercepts注解用于声明当前类是拦截器,其值为@Signature数组,表明要拦截的接口、方法以及对应的参数类型
@Intercepts({@Signature(method="prepare",type=StatementHandler.class,args={Connection.class}),@Signature(method="query",type=StatementHandler.class,args={java.sql.Statement.class,ResultHandler.class})})publicclassTenantInterceptorimplementsInterceptor{.....
例如上面的类声明,第一个Signature标注拦截了StatementHandler类下的入参是一个Connection的名为prepare的方法。
第二个Signature标注拦截StatementHandler类中包含2个入参(分别为Statement和ResultHandler类型)的名为query的方法。
最后,声明的Interceptor需要注册到mybatis的plug中才能生效。
<!--配置mybatis--><beanid="sqlSessionFactory"class="org.mybatis.spring.SqlSessionFactoryBean"><propertyname="dataSource"ref="dataSource"/><propertyname="configLocation"value="classpath:mybatis/mybatis-config.xml"/><!--mapper扫描--><propertyname="mapperLocations"value="classpath:mybatis/*/*.xml"/><propertyname="plugins"><array><!--注册自己的拦截器--><beanid="paginationInterceptor"class="xxx.xxx.TenantInterceptor"></bean></array></property></bean>
关于“mybatis如何实现SQL查询拦截修改”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,使各位可以学到更多知识,如果觉得文章不错,请把它分享出去让更多的人看到。
声明:本站所有文章资源内容,如无特殊说明或标注,均为采集网络资源。如若本站内容侵犯了原著者的合法权益,可联系本站删除。