欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring中使用自定義ThreadLocal存儲(chǔ)導(dǎo)致的坑及解決

 更新時(shí)間:2021年12月06日 16:03:04   作者:javarrr  
這篇文章主要介紹了Spring中使用自定義ThreadLocal存儲(chǔ)導(dǎo)致的坑及解決,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

Spring自定義ThreadLocal存儲(chǔ)導(dǎo)致的坑

Spring 中有時(shí)候我們需要存儲(chǔ)一些和 Request 相關(guān)聯(lián)的變量,例如用戶的登陸有關(guān)信息等,它的生命周期和 Request 相同。

一個(gè)容易想到的實(shí)現(xiàn)辦法是使用ThreadLocal

public class SecurityContextHolder {
    private static final ThreadLocal<SecurityContext> securityContext = new ThreadLocal<SecurityContext>();
    public static void set(SecurityContext context) {
        securityContext.set(context);
    }
    public static SecurityContext get() {
        return securityContext.get();
    }
    public static void clear() {
        securityContext.remove();
    }
}

使用一個(gè)自定義的HandlerInterceptor將有關(guān)信息注入進(jìn)去

@Slf4j
@Component
public class RequestInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws
            Exception {
        try {
            SecurityContextHolder.set(retrieveRequestContext(request));
        } catch (Exception ex) {
            log.warn("讀取請求信息失敗", ex);
        }
        return true;
    }
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable
            ModelAndView modelAndView) throws Exception {
        SecurityContextHolder.clear();
}

通過這樣,我們就可以在 Controller 中直接使用這個(gè) context,很方便的獲取到有關(guān)用戶的信息

@Slf4j
@RestController
class Controller {
  public Result get() {
     long userId = SecurityContextHolder.get().getUserId();
     // ...
  }
}

這個(gè)方法也是很多博客中使用的。然而這個(gè)方法卻存在著一個(gè)很隱蔽的坑: HandlerInterceptor 的 postHandle 并不總是會(huì)調(diào)用。

當(dāng)Controller中出現(xiàn)Exception

@Slf4j
@RestController
class Controller {
  public Result get() {
     long userId = SecurityContextHolder.get().getUserId();
     // ...
     throw new RuntimeException();
  }
}

或者在HandlerInterceptor的preHandle中出現(xiàn)Exception

@Slf4j
@Component
public class RequestInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws
            Exception {
        try {
            SecurityContextHolder.set(retrieveRequestContext(request));
        } catch (Exception ex) {
            log.warn("讀取請求信息失敗", ex);
        }
        // ...
        throw new RuntimeException();
        //...
        return true;
    }
}

這些情況下, postHandle 并不會(huì)調(diào)用。這就導(dǎo)致了 ThreadLocal 變量不能被清理。

在平常的 Java 環(huán)境中,ThreadLocal 變量隨著 Thread 本身的銷毀,是可以被銷毀掉的。但 Spring 由于采用了線程池的設(shè)計(jì),響應(yīng)請求的線程可能會(huì)一直常駐,這就導(dǎo)致了變量一直不能被 GC 回收。更糟糕的是,這個(gè)沒有被正確回收的變量,由于線程池對線程的復(fù)用,可能會(huì)串到別的 Request 當(dāng)中,進(jìn)而直接導(dǎo)致代碼邏輯的錯(cuò)誤。

為了解決這個(gè)問題,我們可以使用 Spring 自帶的 RequestContextHolder ,它背后的原理也是 ThreadLocal,不過它總會(huì)被更底層的 Servlet 的 Filter 清理掉,因此不存在泄露的問題。

下面是一個(gè)使用RequestContextHolder重寫的例子

public class SecurityContextHolder {
    private static final String SECURITY_CONTEXT_ATTRIBUTES = "SECURITY_CONTEXT";
    public static void setContext(SecurityContext context) {
        RequestContextHolder.currentRequestAttributes().setAttribute(
                SECURITY_CONTEXT_ATTRIBUTES,
                context,
                RequestAttributes.SCOPE_REQUEST);
    }
    public static SecurityContext get() {
        return (SecurityContext)RequestContextHolder.currentRequestAttributes()
                .getAttribute(SECURITY_CONTEXT_ATTRIBUTES, RequestAttributes.SCOPE_REQUEST);
    }
}

除了使用 RequestContextHolder 還可以使用 Request Scope 的 Bean,或者使用 ThreadLocalTargetSource ,原理上是類似的。

需要時(shí)刻注意 ThreadLocal 相當(dāng)于線程內(nèi)部的 static 變量,是一個(gè)非常容易產(chǎn)生泄露的點(diǎn),因此使用 ThreadLocal 應(yīng)該額外小心。

Threadlocal可能會(huì)產(chǎn)生內(nèi)存泄露的問題及原理

剛遇到一個(gè)關(guān)于threadlocal的內(nèi)存泄漏問題,剛好總結(jié)一下

比較常用的這里先不提,直接提比較重要的部分

為什么會(huì)產(chǎn)生內(nèi)存泄露?

    public void set(T value) {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null)
            map.set(this, value);
        else
            createMap(t, value);
    }

set方法里面,先調(diào)用到當(dāng)前線程thread,每個(gè)線程里都會(huì)有一個(gè)threadlocals成員變量,指向?qū)?yīng)的ThreadLocalMap ,然后以new出來的引用作為key,和給定的value一塊保存起來。

當(dāng)外部引用解除以后,對應(yīng)的ThreadLocal對象由于被內(nèi)部ThreadLocalMap 引用,不會(huì)GC,可能會(huì)導(dǎo)致內(nèi)存泄露。

JVM解決的辦法

        static class Entry extends WeakReference<ThreadLocal<?>> {
            /** The value associated with this ThreadLocal. */
            Object value;
            Entry(ThreadLocal<?> k, Object v) {
                super(k);
                value = v;
            }
        }

繼承了一個(gè)軟引用,在系統(tǒng)進(jìn)行g(shù)c的時(shí)候就可以回收

但是回收以后,key變成null,value也無法被訪問到,還是可能存在內(nèi)存泄露。 因此一旦不用了,必須對里面的keyvalue對remove掉,否則就會(huì)有內(nèi)存泄露;而且在threadlocal源碼里面,在每次get或者set的時(shí)候會(huì)清楚里面key為value的記錄

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • 新手場景Java線程相關(guān)問題及解決方案

    新手場景Java線程相關(guān)問題及解決方案

    這篇文章主要介紹了新手場景Java線程相關(guān)問題及解決方案,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-07-07
  • Idea中使用Git的流程

    Idea中使用Git的流程

    這篇文章主要介紹了Idea中使用Git的流程,git是目前流行的分布式版本管理系統(tǒng)。本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧
    2020-09-09
  • SpringBoot請求發(fā)送與信息響應(yīng)匹配實(shí)現(xiàn)方法介紹

    SpringBoot請求發(fā)送與信息響應(yīng)匹配實(shí)現(xiàn)方法介紹

    這篇文章主要介紹了SpringBoot請求發(fā)送與信息響應(yīng)匹配實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧
    2022-10-10
  • Trace?在多線程異步體系下傳遞流程解析

    Trace?在多線程異步體系下傳遞流程解析

    這篇文章主要為大家介紹了Trace?在多線程異步體系下傳遞流程解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-12-12
  • springboot實(shí)現(xiàn)執(zhí)行sql語句打印到控制臺(tái)

    springboot實(shí)現(xiàn)執(zhí)行sql語句打印到控制臺(tái)

    這篇文章主要介紹了springboot實(shí)現(xiàn)執(zhí)行sql語句打印到控制臺(tái)的操作,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • Java比較兩個(gè)對象是否相等的方法

    Java比較兩個(gè)對象是否相等的方法

    這篇文章主要介紹了Java比較兩個(gè)對象是否相等的方法,文中給出了三種方法,并通過代碼講解的非常詳細(xì),對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下
    2024-03-03
  • mybatis多對多關(guān)聯(lián)實(shí)戰(zhàn)教程(推薦)

    mybatis多對多關(guān)聯(lián)實(shí)戰(zhàn)教程(推薦)

    下面小編就為大家?guī)硪黄猰ybatis多對多關(guān)聯(lián)實(shí)戰(zhàn)教程(推薦)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-10-10
  • 在啟動(dòng)后臺(tái) jar包時(shí),使用指定的 application.yml操作

    在啟動(dòng)后臺(tái) jar包時(shí),使用指定的 application.yml操作

    這篇文章主要介紹了在啟動(dòng)后臺(tái) jar包時(shí),使用指定的 application.yml操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-10-10
  • Java Spring事務(wù)的隔離級別詳解

    Java Spring事務(wù)的隔離級別詳解

    這篇文章主要介紹了Java Spring事務(wù)的隔離級別,分享了相關(guān)代碼示例,小編覺得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下
    2021-10-10
  • Mybatis攔截器實(shí)現(xiàn)分頁

    Mybatis攔截器實(shí)現(xiàn)分頁

    本文介紹使用Mybatis攔截器,實(shí)現(xiàn)分頁;并且在dao層,直接返回自定義的分頁對象。具有很好的參考價(jià)值,下面跟著小編一起來看下吧
    2017-01-01

最新評論