浅谈ThreadLocal

感觉ThreadLocal这家伙面试必问,但可能使用上确没怎么用,不过其实还是很有用处的。

1.使用方法:

使用如ThreadLocal<String> localVar = new ThreadLocal<>();

有T get()、set(T value)、remove()等方法。

2.Threadlocal如何实现线程安全?:

Threadlocal里面有一个内置静态类ThreadLocalMap,是一个类似HashMap的,

然后在Thread类中有一个ThreadLocal.ThreadLocalMap threadLocals 属性,

每次操作ThreadLocal(get和set)对象时其实是操作的各个线程内部的ThreadLocalMap (key是这个ThreadLocal对象本身,value是值),即使ThreadLocal变量是定义在公共地方,多线程操作这个变量也是相互隔离互不影响的。

3.Threadlocal如何用弱引用WeakReference防止内存泄漏?

ThreadLocal.ThreadLocalMap里面有一个静态内部类Entry,而ThreadLocal.ThreadLocalMap的元素就是Entry,而Entry extends WeakReference<ThreadLocal<?>>,

然后构造方法:

Entry(ThreadLocal<?> k, Object v) {

                super(k);

                value = v;

 }

可见key是一个弱引用,因为key就是Threadlocal对象本身,而弱引用在gc时会直接被回收而不管内存是否够用,那会不会说Threadlocal很快被回收下次再get就为null了?其实不是,Threadlocal定义的地方(new的地方)有一个强引用,只要强引用还在,在堆中Threadlocal的变量就不会被回收,key定义成弱引用,是因为若Threadlocal定义的地方被回收了(所在线程结束了或手工 tl == null),若key还是强引用(且线程还存在),则这个变量还是不能回收的,定义成弱引用才会被回收。

问题: 若key被回收了,那value没被回收不是也会内存泄漏吗? 是的,value也有可能会造成内存泄漏,但value没有其他强引用若也定义成弱引用则下次就get不到了拿到值会是null。所以除非我们知道线程会被结束回收掉(线程被回收,其变量ThreadLocalMap也会被回收就不会有泄漏),否则最好我们在用完Threadlocal直接调下remove()方法释放内存,比如有公司就出现过使用Threadlocal产生内存泄漏最终内存溢出,因为他用的是线程池,线程一直存在不会销毁所以其下ThreadLocalMap的元素会越累越多

4.有应用的地方:

Spring事务实现: spring事务实现就是基于数据库的事务,比如mysql,比如@Transactional是采用AOP 反射实现事务,底层是保证同一个数据库连接来保证事务的,如果同一个线程内多次操作数据库,每次都是从连接池拿不同连接会很浪费性能,所以同一个线程里面都是用同一个数据库连接的,但如何保证同一个数据库连接呢?其实就是用Threadlocal来保证。

最常见的 ThreadLocal 使用场景为用来解决数据库连接、Session 管理等。数据库连接和 Session 管理涉及多个复杂对象的初始化和关闭。如果在每个线程中声明一些私有变量来进行操作,那这个线程就变得不那么“轻量”了,需要频繁的创建和关闭连接。

5.ThreadLocal的一些子类:

ThreadLocal<T>的实现:

NamedThreadLocal:spring的,只是比ThreadLocal多一个name属性。

InheritableThreadLocal: jdk的,能解决父子线程的值传递,但解决不了线程池里的值传递。

NamedInheritableThreadLocal:spring的,只是比InheritableThreadLocal多一个name属性。

TransmittableThreadLocal:alibaba的,继承InheritableThreadLocal,能解决线程池间的值传递,但解决不了并行流里的值传递。

6.说下并行流parallelStream

并行流是基于forkjoin。工作被拆成细小的worker,不会阻塞线程,性能要比普通线程池高,但无法传递threadLocal的值。

那如何解决并行流里的值传递呢?

解决办法:

使用TransmittableThreadLocal,并且,

在启动的VM参数加上:

-javaagent:D:/programfiles/apache-maven-3.6.3/m2/myrepository/com/alibaba/transmittable-thread-local/2.12.2/transmittable-thread-local-2.12.2.jar

7.TransmittableThreadLocal保存用户上下文例子:

当然,除了保存用户信息,还可用作保存权限相关信息,国际化语言环境信息等(一般通过cookie传入)。

①一个UserContext类

private static final TransmittableThreadLocal<User> USER_INFO = new TransmittableThreadLocal<>();

public static void setUser(User user){

        USER_INFO.remove();

        USER_INFO.set(user);

    }

public static User getUser(){

        return USER_INFO.get();

    }

    public static void clear(){

        USER_INFO.remove();

    }

②自定义一个Filter初始化ttl

public class MyFilter extends OncePerRequestFilter

在其下doFilterInternal()方法中执行

UserContext.setUser(user);【当然user是根据token拿到的】

(这个MyFilter 可以在

@Configuration

public class SecurityConfig extends WebSecurityConfigurerAdapter

中执行

http.addFilterBefore(new MyFilter (), UsernamePasswordAuthenticationFilter.class);

或者直接将MyFilter 定义成bean

两种方法都能使得MyFilter 生效。

)

③自定义拦截器,在请求结束后清除ttl变量。(一般对ttl建议写下清除好习惯更为保障,避免内存泄漏情况)

@Component

public class AuthInterceptor implements AsyncHandlerInterceptor {

    /**

     * 请求结束,清除TTL中的身份信息

     */

    @Override

    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex){

        UserContext.clear();

    }

}

注册拦截器:

@Configuration

public class AppConfiguration implements WebMvcConfigurer {

    @Autowired

    private AuthInterceptor authInterceptor;

    @Override

    public void addInterceptors(InterceptorRegistry registry) {

        registry.addInterceptor(authInterceptor);

    }

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
修改ThreadLocal的值可以通过以下步骤实现: 1. 首先,创建一个ThreadLocal对象,并将其初始化为需要修改的值。可以使用ThreadLocal的`set()`方法来设置值。例如,`threadLocal.set(value)`,其中value是要设置的新值。 2. 接下来,可以在需要修改ThreadLocal值的线程中使用上述设置的ThreadLocal对象。可以通过`threadLocal.get()`方法来获取该线程的ThreadLocal值。 3. 然后,对获取的ThreadLocal值进行修改,可以使用任何适当的方法来更改值。 4. 最后,使用ThreadLocal的`set()`方法将修改后的值设置回ThreadLocal对象中。例如,`threadLocal.set(modifiedValue)`,其中modifiedValue是修改后的值。 需要注意的是,每个线程都有自己独立的ThreadLocal副本,因此在不同的线程中修改ThreadLocal值不会相互影响。这是ThreadLocal的特性之一,使得每个线程可以独立地管理自己的数据。 总结起来,修改ThreadLocal的值需要创建ThreadLocal对象、设置新值、获取当前线程的ThreadLocal值、修改该值,最后将修改后的值设置回ThreadLocal对象中。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [浅谈ThreadLocal](https://blog.csdn.net/qq_36609994/article/details/121163994)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *3* [ThreadLocal](https://blog.csdn.net/weixin_43573186/article/details/125068926)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值