91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Spring中使用自定義ThreadLocal存儲導致的坑怎么解決

發布時間:2022-03-14 16:19:43 來源:億速云 閱讀:213 作者:iii 欄目:web開發

這篇文章主要介紹了Spring中使用自定義ThreadLocal存儲導致的坑怎么解決的相關知識,內容詳細易懂,操作簡單快捷,具有一定借鑒價值,相信大家閱讀完這篇Spring中使用自定義ThreadLocal存儲導致的坑怎么解決文章都會有所收獲,下面我們一起來看看吧。

  Spring 中有時候我們需要存儲一些和 Request 相關聯的變量,例如用戶的登陸有關信息等,它的生命周期和 Request 相同。一個容易想到的實現辦法是使用 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();

  }

  }

  使用一個自定義的 HandlerInterceptor 將有關信息注入進去:

  @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 中直接使用這個 context,很方便的獲取到有關用戶的信息:

  @Slf4j

  @RestController

  class Controller {

  public Result get() {

  long userId = SecurityContextHolder.get()。getUserId();

  // &hellip;

  }

  }

  這個方法也是很多博客中使用的。然而這個方法卻存在著一個很隱蔽的坑: HandlerInterceptor 的 postHandle 并不總是會調用。

  當 Controller 中出現 Exception:

  @Slf4j

  @RestController

  class Controller {

  public Result get() {

  long userId = SecurityContextHolder.get()。getUserId();

  // &hellip;

  throw new RuntimeException();

  }

  }

  或者在 HandlerInterceptor 的 preHandle 中出現 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);

  }

  // &hellip;

  throw new RuntimeException();

  //&hellip;

  return true;

  }

  }

  這些情況下, postHandle 并不會調用。這就導致了 ThreadLocal 變量不能被清理。

  在平常的 Java 環境中,ThreadLocal 變量隨著 Thread 本身的銷毀,是可以被銷毀掉的。但 Spring 由于采用了線程池的設計,響應請求的線程可能會一直常駐,這就導致了變量一直不能被 GC 回收。更糟糕的是,這個沒有被正確回收的變量,由于線程池對線程的復用,可能會串到別的 Request 當中,進而直接導致代碼邏輯的錯誤。

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

  下面是一個使用 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 ,原理上是類似的。

  需要時刻注意 ThreadLocal 相當于線程內部的 static 變量,是一個非常容易產生泄露的點,因此使用 ThreadLocal 應該額外小心。

關于“Spring中使用自定義ThreadLocal存儲導致的坑怎么解決”這篇文章的內容就介紹到這里,感謝各位的閱讀!相信大家對“Spring中使用自定義ThreadLocal存儲導致的坑怎么解決”知識都有一定的了解,大家如果還想學習更多知識,歡迎關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

江西省| 将乐县| 泸西县| 贵州省| 枣庄市| 辽中县| 顺平县| 鄂托克旗| 邢台县| 红河县| 溧水县| 永仁县| 龙南县| 雷山县| 砀山县| 那曲县| 长治县| 拜泉县| 临桂县| 离岛区| 舟山市| 驻马店市| 开阳县| 开远市| 普宁市| 衡阳市| 湘乡市| 民县| 内黄县| 铁岭市| 清涧县| 巴南区| 阳城县| 三江| 剑河县| 太和县| 乌鲁木齐市| 南丰县| 杭锦后旗| 盐城市| 襄樊市|