Mybatis Interceptor 经典场景

1.Mybatis Interceptor 的应用场景很多,比如sql语句动态生成,resultMap动态生成

下面就用代码来分析下resultMap动态生成吧,这个是ORM框架常用的功能哦

 1 package com.xx.transjob.common.db;
 2 
 3 import com.baomidou.mybatisplus.annotation.TableName;
 4 import org.apache.ibatis.cache.CacheKey;
 5 import org.apache.ibatis.executor.Executor;
 6 import org.apache.ibatis.mapping.BoundSql;
 7 import org.apache.ibatis.mapping.MappedStatement;
 8 import org.apache.ibatis.mapping.ResultMap;
 9 import org.apache.ibatis.mapping.ResultMapping;
10 import org.apache.ibatis.plugin.Interceptor;
11 import org.apache.ibatis.plugin.Intercepts;
12 import org.apache.ibatis.plugin.Invocation;
13 import org.apache.ibatis.plugin.Signature;
14 import org.apache.ibatis.session.ResultHandler;
15 import org.apache.ibatis.session.RowBounds;
16 import org.springframework.stereotype.Component;
17 import org.springframework.util.ClassUtils;
18 import org.springframework.util.CollectionUtils;
19 import org.springframework.util.ReflectionUtils;
20 
21 import java.lang.reflect.Field;
22 import java.util.ArrayList;
23 import java.util.Collection;
24 import java.util.Collections;
25 import java.util.List;
26 
27 @Component
28 @Intercepts({
29         @Signature(type = Executor.class, method = "query",
30                 args = { MappedStatement.class, Object.class,
31                         RowBounds.class, ResultHandler.class }),
32         @Signature(type = Executor.class, method = "query",
33                 args = { MappedStatement.class, Object.class, RowBounds.class,
34                         ResultHandler.class, CacheKey.class, BoundSql.class }),
35 })
36 public class ResultInterceptor implements Interceptor {
37     @Override
38     public Object intercept(Invocation invocation) throws Throwable {
39         if (!(invocation.getTarget() instanceof Executor)) {
40             return invocation.proceed();
41         }
42         MappedStatement ms = (MappedStatement) invocation.getArgs()[0];
43         if (ms.getResource().contains(".xml")) {
44             return invocation.proceed();
45         }
46         ResultMap resultMap = ms.getResultMaps().iterator().next();
47         if (!CollectionUtils.isEmpty(resultMap.getResultMappings())) {
48             return invocation.proceed();
49         }
50         Class<?> mapType = resultMap.getType();
51         if (ClassUtils.isAssignable(mapType, Collection.class)
52             ||mapType.getAnnotation(TableName.class)==null) {
53             return invocation.proceed();
54         }
55         //根据返回类型查找model类型
56         DbUtils.ModelClassInfo modelClassInfo=DbUtils.getModelClassInfo(mapType);
57         List<ResultMapping> resultMappings = new ArrayList<>(modelClassInfo.modelFields.size());
58         //根据model动态生成resultMap
59         for(DbUtils.ModelFieldInfo fieldInfo:modelClassInfo.modelFields){
60             ResultMapping resultMapping = new ResultMapping.Builder(ms.getConfiguration(),
61                     fieldInfo.field.getName(),
62                     fieldInfo.column, fieldInfo.field.getType()).build();
63             resultMappings.add(resultMapping);
64         }
65         ResultMap resultMapReal=new ResultMap.Builder(ms.getConfiguration(),resultMap.getId(),mapType,resultMappings,true).build();
66         //替换原来的resultMaps
67         Field field = ReflectionUtils.findField(MappedStatement.class, "resultMaps");
68         ReflectionUtils.makeAccessible(field);
69         ReflectionUtils.setField(field, ms, Collections.singletonList(resultMapReal));
70         return invocation.proceed();
71     }
72 }

 

posted @ 2023-08-15 10:11  dint  阅读(24)  评论(0编辑  收藏  举报