设为首页 收藏本站
查看: 1042|回复: 0

[经验分享] 深入浅出Mybatis-改造Cache

[复制链接]

尚未签到

发表于 2016-11-27 09:01:04 | 显示全部楼层 |阅读模式
转自 http://blog.csdn.net/hupanfeng/article/details/16950161
 
为了方便修改BUG,我在github上创建了一个仓库,地址:https://github.com/hupanfeng/hdd。欢迎大家在留言里提交问题,我会尽快修复,并将修复的代码提交至github上。
 
在前面的文章里,我开发了两个插件:根据注解实现的sql自动生成插件和分页插件。这两个插件在没有开启cache的情况下可以很好的使用,但开启cache后却出现了一些问题,为了解决这些问题,我编写了拦截cache的插件,通过这个拦截器修正了这些问题。
问题
什么问题
最容易出现的问题是开启cache后,分页查询时无论查询哪一页都返回第一页的数据。另外,使用sql自动生成插件生成get方法的sql时,传入的参数不起作用,无论传入的参数是多少,都返回第一个参数的查询结果。
 
为什么出现这些问题
在之前讲解Mybatis的执行流程的时候提到,在开启cache的前提下,Mybatisexecutor会先从缓存里读取数据,读取不到才去数据库查询。问题就出在这里,sql自动生成插件和分页插件执行的时机是在statementhandler里,而statementhandler是在executor之后执行的,无论sql自动生成插件和分页插件都是通过改写sql来实现的,executor在生成读取cachekeykeysql以及对应的参数值构成)时使用都是原始的sql,这样当然就出问题了。
解决问题
找到问题的原因后,解决起来就方便了。只要通过拦截器改写executor里生成key的方法,在生成可以时使用自动生成的sql(对应sql自动生成插件)或加入分页信息(对应分页插件)就可以了。
拦截器签名
 
[java] view plaincopy 



  • @Intercepts({@Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})})  
  • public class CacheInterceptor implements Interceptor {  
  • ...  
  • }  

 
从签名里可以看出,要拦截的目标类型是Executor(注意:type只能配置成接口类型),拦截的方法是名称为query的方法。
intercept的实现
[java] view plaincopy 



  • public Object intercept(Invocation invocation) throws Throwable {  
  •         Executor executorProxy = (Executor) invocation.getTarget();  
  •         MetaObject metaExecutor = MetaObject.forObject(executorProxy, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY);  
  •         // 分离代理对象链  
  •         while (metaExecutor.hasGetter("h")) {  
  •             Object object = metaExecutor.getValue("h");  
  •             metaExecutor = MetaObject.forObject(object, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY);  
  •         }  
  •         // 分离最后一个代理对象的目标类  
  •         while (metaExecutor.hasGetter("target")) {  
  •             Object object = metaExecutor.getValue("target");  
  •             metaExecutor = MetaObject.forObject(object, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY);  
  •         }  
  •         Object[] args = invocation.getArgs();  
  •         return this.query(metaExecutor, args);  
  •     }  
  •   
  •     public <E> List<E> query(MetaObject metaExecutor, Object[] args) throws SQLException {  
  •         MappedStatement ms = (MappedStatement) args[0];  
  •         Object parameterObject = args[1];  
  •         RowBounds rowBounds = (RowBounds) args[2];  
  •         ResultHandler resultHandler = (ResultHandler) args[3];  
  •         BoundSql boundSql = ms.getBoundSql(parameterObject);  
  •         // 改写key的生成  
  •         CacheKey cacheKey = createCacheKey(ms, parameterObject, rowBounds, boundSql);  
  •         Executor executor = (Executor) metaExecutor.getOriginalObject();  
  •         return executor.query(ms, parameterObject, rowBounds, resultHandler, cacheKey, boundSql);  
  •     }  
  •   
  •     private CacheKey createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql) {  
  •         Configuration configuration = ms.getConfiguration();  
  •         pageSqlId = configuration.getVariables().getProperty("pageSqlId");  
  •         if (null == pageSqlId || "".equals(pageSqlId)) {  
  •             logger.warn("Property pageSqlId is not setted,use default '.*Page$' ");  
  •             pageSqlId = defaultPageSqlId;  
  •         }  
  •         CacheKey cacheKey = new CacheKey();  
  •         cacheKey.update(ms.getId());  
  •         cacheKey.update(rowBounds.getOffset());  
  •         cacheKey.update(rowBounds.getLimit());  
  •         List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();  
  •         // 解决自动生成SQL,SQL语句为空导致key生成错误的bug  
  •         if (null == boundSql.getSql() || "".equals(boundSql.getSql())) {  
  •             String id = ms.getId();  
  •             id = id.substring(id.lastIndexOf(".") + 1);  
  •             String newSql = null;  
  •             try {  
  •                 if ("select".equals(id)) {  
  •                     newSql = SqlBuilder.buildSelectSql(parameterObject);  
  •                 }  
  •                 SqlSource sqlSource = buildSqlSource(configuration, newSql, parameterObject.getClass());  
  •                 parameterMappings = sqlSource.getBoundSql(parameterObject).getParameterMappings();  
  •                 cacheKey.update(newSql);  
  •             } catch (Exception e) {  
  •                 logger.error("Update cacheKey error.", e);  
  •             }  
  •         } else {  
  •             cacheKey.update(boundSql.getSql());  
  •         }  
  •   
  •         MetaObject metaObject = MetaObject.forObject(parameterObject, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY);  
  •   
  •         if (parameterMappings.size() > 0 && parameterObject != null) {  
  •             TypeHandlerRegistry typeHandlerRegistry = ms.getConfiguration().getTypeHandlerRegistry();  
  •             if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {  
  •                 cacheKey.update(parameterObject);  
  •             } else {  
  •                 for (ParameterMapping parameterMapping : parameterMappings) {  
  •                     String propertyName = parameterMapping.getProperty();  
  •                     if (metaObject.hasGetter(propertyName)) {  
  •                         cacheKey.update(metaObject.getValue(propertyName));  
  •                     } else if (boundSql.hasAdditionalParameter(propertyName)) {  
  •                         cacheKey.update(boundSql.getAdditionalParameter(propertyName));  
  •                     }  
  •                 }  
  •             }  
  •         }  
  •         // 当需要分页查询时,将page参数里的当前页和每页数加到cachekey里  
  •         if (ms.getId().matches(pageSqlId) && metaObject.hasGetter("page")) {  
  •             PageParameter page = (PageParameter) metaObject.getValue("page");  
  •             if (null != page) {  
  •                 cacheKey.update(page.getCurrentPage());  
  •                 cacheKey.update(page.getPageSize());  
  •             }  
  •         }  
  •         return cacheKey;  
  • }   

plugin的实现
[java] view plaincopy 



  • public Object plugin(Object target) {  
  •     // 当目标类是CachingExecutor类型时,才包装目标类,否者直接返回目标本身,减少目标被代理的  
  •     // 次数  
  •     if (target instanceof CachingExecutor) {  
  •         return Plugin.wrap(target, this);  
  •     } else {  
  •         return target;  
  •     }  
  • }  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-306007-1-1.html 上篇帖子: Mybatis源码研究6:元数据(metadata) 下篇帖子: ibatis又名(mybatis)使用方法,针对jdbc的封装
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表