项目场景:
最早是想用Mybatis拦截器/插件 拦截结果集并修改!
看看文档和网上的博客,了解了大概也写了个demo;demo使用的时候没有问题所以没有深入研究
Mybatis 拦截器/插件 之前使用new的方式配置到配置类中 使拦截器生效 后来改成注入的方式的时候出现了问题
问题描述
问题 有点不好描述
Mybatis 拦截器/插件 之前使用new的方式配置到配置类中 使拦截器生效 后来改成注入的方式的时候出现了问题
之前的配置类如下:
@Configuration
public class MapperConfig {
@Autowired
private SqlSessionFactory sqlSessionFactory;
@PostConstruct
public void myInterceptor(){
sqlSessionFactory.getConfiguration().addInterceptor(new ResultSetForMapperInterceptor());
}
}
@Component
@Intercepts({@Signature(type = ResultSetHandler.class,method = "handleResultSets",args = Statement.class)
})
public class ResultSetForMapperInterceptor implements Interceptor {
private static Logger logger= LogManager.getLogger(ResultSetForMapperInterceptor.class);
private final static String METHOD_NAME = "com.yunjian.build.codegen.mapper.GenFormConfMapper.selectCountGroup";
@Override
public Object intercept(Invocation invocation) throws Throwable {
logger.info("----------mybatis 拦截器start----------");
logger.info("----------mybatis 拦截器end----------");
return resultObj;
}
@Override
public Object plugin(Object target) {
System.out.println("plugin 方法执行^^^^^^^^^^^^^^");
return Plugin.wrap(target,this);
}
@Override
public void setProperties(Properties properties) {
logger.debug(properties);
}
}
后来添加将拦截器放入ioc容器进行管理的代码如下:
@Configuration
public class MapperConfig {
@Autowired
private SqlSessionFactory sqlSessionFactory;
@Autowired
private ResultSetForMapperInterceptor resultSetForMapperInterceptor;
@PostConstruct
public void myInterceptor(){
sqlSessionFactory.getConfiguration().addInterceptor(resultSetForMapperInterceptor);
}
}
@Component
@Intercepts({@Signature(type = ResultSetHandler.class,method = "handleResultSets",args = Statement.class)
})
public class ResultSetForMapperInterceptor implements Interceptor {
private static Logger logger= LogManager.getLogger(ResultSetForMapperInterceptor.class);
private final static String METHOD_NAME = "com.yunjian.build.codegen.mapper.GenFormConfMapper.selectCountGroup";
@Autowired
private DictComponent dictComponent;
@Override
public Object intercept(Invocation invocation) throws Throwable {
logger.info("----------mybatis 拦截器start----------");
String dictName = dictComponent.getDictNameByParam("log_type", "9");
System.out.println("------ dictName -----"+dictName);
logger.info("----------mybatis 拦截器end----------");
return resultObj;
}
@Override
public Object plugin(Object target) {
System.out.println("plugin 方法执行^^^^^^^^^^^^^^");
return Plugin.wrap(target,this);
}
@Override
public void setProperties(Properties properties) {
logger.debug(properties);
}
}
其实不用IOC来管理拦截器也能用 但拦截器需要注入其他实例的时候会报错,因为拦截器是new的不是被ioc管理的 所以拦截器此时获取不到ioc内的实例;最终改成了将拦截器注入ioc的方式;
原因分析:
断点调试后发现,拦截器的
invocation.getTarget();
执行了两次 第二次执行很诡异,方法执行完之后跳回到getTarget() ;
最终我也没有找到问题是哪里;因为 拦截器没有重复构造,而且配置类也没有重复添加拦截器;断点调试方法执行完只跳回getTarget方法;但从断点调试的结果看,问题大概是出在注解上;
这里标记一下 其实解决这个问题很容易 但是并没搞懂这个的原理;
解决方案:
去掉
@PostConstruct
注解 就解决了
这个注解是为了让方法提前运行,方法上加打印,会看到启动服务器前期就会执行@PostConstruct标注的方法;其他细节还没有深入了解;这里标记一下这个注解,有空仔细看看;
其他:
@Autowired(false)
Autowired注解默认传入true 检查bean是否存在不存在就会报错;而传入false不会检查bean在容器中是否存在;没有就不注入