Dubbo的filter中使用ThreadLocal引发的坑

在filter中使用ThreadLocal要特别注意一些问题,首先ThreadLocal是跟着线程走的,而不管是dubbo或者其他的rpc框架或者Springmvc都有个特点:使用的是线程池模型,当线程执行任务结束之后会回到线程池,这时如果在回到线程池之前ThreadLocal没有被清理,当下一次请求拿到这个线程的时候还能读取到之前没有被清理的ThreadLocal的数据,这样显然不是我们想要的结果了。
处理方式以dubbo的filter为例:

//在服务端的filter中invoker.invoke(invocation)执行后清除ThreadLocal变量
invoker.invoke(invocation);
//TODO 此处清理ThreadLocal的信息

看着是不是感觉没啥问题了?其实还有坑在里面,当服务中出现内部调用的时候,如aaaService里面调用bbbService,而aaaService和bbbService都属于同一个应用的时候,他们的调用不会走dubbo协议,而是直接走内部调用,但是会把filter走一遍,这时候因为不管调用aaaService还是bbbService都是同一个线程在执行,filter走一遍之后ThreadLocal被清理了,这种情况下并不是我们想要的了。。因为后续的逻辑里可能我们还需要用到这个ThreadLocal,所以我们在此加入一个判断,用于判断是否为内部调用。此处我已经找到了对应的处理方式。代码如下

//在服务端的filter中invoker.invoke(invocation)执行后清除ThreadLocal变量
invoker.invoke(invocation);
// injvm是dubbo的内部协议,内部调用伪协议,我们只有不等于这个协议的时候才需要清理ThreadLocal信息
if(!"injvm".equals(invoker.getUrl().getProtocol())){
	//TODO 此处清理ThreadLocal的信息
}
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值