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

使用springboot單例模式與線程安全問題踩的坑

 更新時(shí)間:2021年08月04日 10:10:55   作者:會飛的基德  
這篇文章主要介紹了使用springboot單例模式與線程安全問題踩的坑,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

springboot單例模式與線程安全問題踩的坑

最近有客戶反映,使用公司產(chǎn)品時(shí),偶爾會存在崩潰情況,自己測試無問題,然后去查日志,是報(bào)空指針。

于是順藤摸瓜 往上找,好嘛,之前的開發(fā)使用了成員變量,感覺問題就是在這里了,因?yàn)楸娝苤?,springboot 采用的是單例模式,所以,使用成員變量時(shí)一定要謹(jǐn)慎。

下面上一張?jiān)擃惖慕貓D:

大家可能看到了,該類上面加上了@Scope("prototype") 注解,該注解的作用是將該類變成多例模式。講道理因?yàn)樽優(yōu)榱硕嗬?,?yīng)該不會有線程問題了。

我先說下我這邊的一個(gè)代碼環(huán)境,上面大家看到的BaseController這個(gè)類里面有個(gè)init方法,會在繼承它的類的所有方法前執(zhí)行。

使用的是@ModelAttribute注解,這個(gè)注解的意思是,在該controller的所有方法前執(zhí)行,意在初始化,我猜測之前的同事應(yīng)該是為了獲取相同的一些參數(shù),抽調(diào)出來做一個(gè)父類,隨著迭代,別的同事為了方便,拿來就用,導(dǎo)致很多controller繼承了該類。

@Scope("prototype")注解:

大家設(shè)想一下,若父類加了@Scope("prototype")注解,子類controller并沒有加該注解,會怎樣呢?該注解是否還有意義?再比如,我在某service上加上@Scope("prototype")注解,但調(diào)用的controller沒有加@Scope("prototype")注解,那么會出現(xiàn)什么樣的結(jié)果呢?大家可以去測試一下,測試方法也很簡單,就是在對應(yīng)的父類或service的無參構(gòu)造方法里打印該類的地址。

下面說下我的測試結(jié)果:

先說父類上加了@Scope("prototype")注解,子類上沒有加這種情況。結(jié)果是,同一子類繼承的為同一父類,不同子類繼承為不同父類。理解一下,很簡單,因?yàn)閟pringboot為單例模式,所以子類為單例,那么只有一個(gè)子類,父類肯定是一樣的。所以,不同線程過來使用的為同一變量,就會有問題。

同理:

在service上標(biāo)注@Scope("prototype")注解,那在同一個(gè)controller里,該service還是同一個(gè),也就是說還是單例的,在不同的controller里 是不同的。測試方法同上。

現(xiàn)在說下解決方法:

1、是在繼承該controller的子類上都加上@Scope("prototype")注解。這樣做的好處是簡單。壞處也同樣明顯,因?yàn)槭嵌嗬?,那么就會產(chǎn)生大量的實(shí)體類,占用大量內(nèi)存,若是回收不及時(shí),有可能會出現(xiàn)內(nèi)存溢出。

2、是將變量私有化,比如使用線程變量,對變量加鎖等,技術(shù)上會復(fù)雜一些,而且調(diào)試不太好調(diào)試。說不定那些地方就會出現(xiàn)問題,畢竟是老代碼。

3、將該類轉(zhuǎn)換為攔截器,將變量放入request里,用的時(shí)候取出來。

SpringMVC 或 SpringBoot 默認(rèn)是單例模式(Singleton)

多個(gè)請求是訪問的同一個(gè)方法,是如何實(shí)現(xiàn)線程安全的?

SpringMVC Controller默認(rèn)情況下是Singleton(單例)的,當(dāng)request過來,不用每次創(chuàng)建Controller,會用原來的instance去處理。那么當(dāng)多個(gè)線程調(diào)用它的時(shí)候,會不會發(fā)生線程不安全呢?

1、先說明下 Controller默認(rèn)情況 單例的問題:

使用Spring MVC有一段時(shí)間了,之前一直使用Struts2,在struts2中action都是原型(prototype)的, 說是因?yàn)榫€程安全問題,對于Spring MVC中bean默認(rèn)都是(singleton)單例的,那么用@Controller注解標(biāo)簽注入的Controller類是單例實(shí)現(xiàn)的?

測試結(jié)果發(fā)現(xiàn)spring3中的controller默認(rèn)是單例的,若是某個(gè)controller中有一個(gè)私有的變量i,所有請求到同一個(gè)controller時(shí),使用的i變量是共用的,即若是某個(gè)請求中修改了這個(gè)變量a,則,在別的請求中能夠讀到這個(gè)修改的內(nèi)容。 若是在@Controller之前增加@Scope(“prototype”),就可以改變單例模式為多例模式

以下是測試步驟,代碼與結(jié)果.

1. 如果是單例類型類的,那么在Controller類中的類變量應(yīng)該是共享的,如果不共享,就說明Controller類不是單例。

以下是測試代碼:

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
 
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
@Controller
public class ExampleAction {
    private int singletonInt=1;
     @RequestMapping(value = "/test")
     @ResponseBody
     public String singleton(HttpServletRequest request,
             HttpServletResponse response) throws Exception {
         String data=request.getParameter("data");
         if(data!=null&&data.length()>0){
             try{
              int paramInt= Integer.parseInt(data);
             singletonInt = singletonInt + paramInt;
             }
             catch(Exception ex){
                 singletonInt+=10;
             }
         }else{
             singletonInt+=1000;
         }
         return String.valueOf(singletonInt);
    }
}

分別三次請求: http://localhost:8080/example/test.do?data=15

得到的返回結(jié)果如下。

第一次: singletonInt=15

第二次: singletonInt=30

第三次: singletonInt=45

從以上結(jié)果可以得知,singletonInt的狀態(tài)是共享的,因此Controller是單例的。

2、對別Struts與springmvc對比

Struts2:默認(rèn)prototype,Struts2 是基于類的,處于線程安全的考慮,采用了prototype模式,也就是說每次請求都會新建一個(gè)類來處理,自然就沒有線程安全問題了,每次請求的類和數(shù)據(jù)都是單獨(dú)的。

Springmvc:默認(rèn)singleton 單例模式,Springmvc 是基于方法的,同一個(gè)url的請求是同一個(gè)實(shí)例處理的。每次請求都會把請求參數(shù)傳遞到同一個(gè)方法中,此時(shí)如果類里面有成員變量,那么這個(gè)變量就不是線程安全的了(例如上面的例子 private int singletonInt=1; 這個(gè)變量如果想線程安全則可以用ThreadLocal)。

在類中沒有成員變量的前提下則是線程安全的。

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

相關(guān)文章

  • Spring框架AOP面向切面編程原理全面分析

    Spring框架AOP面向切面編程原理全面分析

    這篇文章主要介紹了Spring框架AOP面向切面編程的全面分析,文中附含詳細(xì)的示例代碼分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2021-09-09
  • MyEclipse配置JDK的全過程

    MyEclipse配置JDK的全過程

    這篇文章主要介紹了MyEclipse配置JDK的全過程,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-04-04
  • springboot+vue實(shí)現(xiàn)驗(yàn)證碼功能

    springboot+vue實(shí)現(xiàn)驗(yàn)證碼功能

    這篇文章主要為大家詳細(xì)介紹了springboot+vue實(shí)現(xiàn)驗(yàn)證碼功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • springboot中實(shí)現(xiàn)上傳文件的功能簡單示例

    springboot中實(shí)現(xiàn)上傳文件的功能簡單示例

    這篇文章主要給大家介紹了關(guān)于springboot中實(shí)現(xiàn)上傳文件功能的相關(guān)資料,在Spring Boot中實(shí)現(xiàn)文件上傳下載功能相對簡單,文中給出了代碼示例,需要的朋友可以參考下
    2023-09-09
  • Java8新特性之新日期時(shí)間庫的使用教程

    Java8新特性之新日期時(shí)間庫的使用教程

    這篇文章主要給大家介紹了關(guān)于Java8新特性之新日期時(shí)間庫使用的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • Java實(shí)現(xiàn)簡單的抽牌游戲

    Java實(shí)現(xiàn)簡單的抽牌游戲

    這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)簡單的抽牌游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-04-04
  • java讀取文件內(nèi)容為string字符串的方法

    java讀取文件內(nèi)容為string字符串的方法

    今天小編就為大家分享一篇java讀取文件內(nèi)容為string字符串的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-07-07
  • java之a(chǎn)ssert關(guān)鍵字用法案例詳解

    java之a(chǎn)ssert關(guān)鍵字用法案例詳解

    這篇文章主要介紹了java之a(chǎn)ssert關(guān)鍵字用法案例詳解,本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-08-08
  • mybatisplus?@Select注解中拼寫動態(tài)sql異常問題的解決

    mybatisplus?@Select注解中拼寫動態(tài)sql異常問題的解決

    這篇文章主要介紹了mybatisplus?@Select注解中拼寫動態(tài)sql異常問題的解決,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • 詳解Spring AOP的實(shí)現(xiàn)方式

    詳解Spring AOP的實(shí)現(xiàn)方式

    AOP是一種思想,是對某一類事情的集中處理,切面就是指某一類特定的問題,所以AOP可以理解為面向特定方法編程,這篇文章主要介紹了Spring AOP的實(shí)現(xiàn)方式,需要的朋友可以參考下
    2024-02-02

最新評論