ThreadLocal 可能踩到的坑

背景

  • 首先ThreadLocal 不做太多的介绍,在多线程场景下有着广泛的应用。
  • 最近在实现OneLog方法级日志的时候,使用了 ThreadLocal 作为缓存和计数器,在调试过程中,发现有一些场景会出现数据错乱和内存溢出的问题。

详情

  • OneLog方法级日志使用 ThreadLocal 有两个场景,一个是将方法入参信息缓存起来,然后在方法结束之后与返回结果进行组装。另一个是作为计数器使用,保证循环方法的场景下,不会打印太多的日志。
  • 实际使用的时候发现了问题,ThreadLocal 对象的生命周期是依赖于线程的,那么对于一个web应用来说,tomcat的线程是循环使用的,也就是说在web应用中,主线程下 ThreadLocal 对象永远不会被回收!
  • 下面是问题复现示例代码:
@Controller
public class TestThreadLocalController {
    //计数器
    private final ThreadLocal<Integer> counter = new ThreadLocal<Integer>() {
        @Override
        protected Integer initialValue() {
            return 0;
        }
    };
    @RequestMapping("/testThreadLocal")
    public @ResponseBody  Integer testThreadLocal() {
        doSomething();
        return counter.get();
    }
    private void doSomething() {
        counter.set(counter.get()+1);
    }
}
  • 执行多次web接口调用,当tomcat线程出现重复使用的时候,即可看到返回值大于1的情况:
    ​​image.png

解决方案

  • 在线程池(包括web主线程)中使用 ThreadLocal 的时候,一定要考虑到 ThreadLocal 对象的生命周期是跟随线程的,会随着线程池的循环而一直存在。
  • 那么在这种场景下,使用 ThreadLocal 就要注意这几点:

    • Map、List、Set等集合对象,要考虑数据是否会一直增加,如果一直增加而不移除的话就会造成内存溢出。
    • 用作计数器的话,使用之后要清0,不然计数的数值会随着线程一直保存下来。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值