Mybatis 拦截器/插件 内注入bean的问题解决

项目场景:

最早是想用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在容器中是否存在;没有就不注入

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值