背景:今天寫代碼遇到一個Controller 中的線程安全問題,那么Spring 的Controller 是單例還是多例的呢?若為單例又如何保證并發安全呢?
一、面試回答
Spring管理的Controller,即加入@Controller 注入的類,默認是單例的,因此建議:
1、不要在Controller 中定義成員變量;(單例非線程安全,會導致屬性重復使用)
2、若必須要在Controller 中定義一個非靜態成員變量,則通過注解@Scope("prototype"),將其設置為多例模式。
二、驗證Controller 單例
驗證代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
|
package com.ausclouds.bdbsec.tjt; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody; /** * @author tjt * @time 2020-08-25 * @desc 驗證Controller 單例 */ @Controller @ResponseBody @RequestMapping ( "/tjt" ) public class TestSingleController { private long money = 10 ; @GetMapping ( "/test1" ) public long testSingleOne(){ money = ++money; System.out.println( "/tjt/test1: the money I have: " + money); return money; } @GetMapping ( "test2" ) public long testSingleTwo(){ money = ++money; System.out.println( "/tjt/test2: the money I have: " + money); return money; } } |
首先,訪問http://localhost:8088/test1
,得到的答案是11
;
接著,再訪問http://localhost:8088/test2
,得到的答案是 12
;
不難看出:同一個變量,兩次訪問得到不同的結果,很明顯是線程不安全的。
驗證截圖:
三、Controller 如何實現多例?
盡量不要在Controller 中定義成員變量,若必須要在Controller 中定義一個非靜態成員變量,則通過注解@Scope("prototype"),將其設置為多例模式;或者是在Controller 中使用ThreadLocal 變量。
驗證代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
|
package com.ausclouds.bdbsec.tjt; import org.springframework.context.annotation.Scope; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody; /** * @author tjt * @time 2020-08-25 * @desc 驗證Controller 單例 */ @Controller @ResponseBody @Scope ( "prototype" ) // 將Controller 設置為多例模式 @RequestMapping ( "/tjt" ) public class TestSingleController { private long money = 10 ; @GetMapping ( "/test1" ) public long testSingleOne(){ money = ++money; System.out.println( "/tjt/test1: after use @Scope the money I have: " + money); return money; } @GetMapping ( "test2" ) public long testSingleTwo(){ money = ++money; System.out.println( "/tjt/test2: after use @Scope the money I have: " + money); return money; } } |
在加上@Scope("prototype")后首先,訪問http://localhost:8088/test1
,得到的答案是11
;
接著,再訪問http://localhost:8088/test2
,得到的答案也是 11
;
不難看出:同一個變量,兩次訪問得到相同的結果。
驗證截圖:
四、作用域
其實,spring bean 的作用域除了上面使用的prototype 外,還有singleton、request、session 和global session 四種;其中request、session 和global session 主要運用在Web 項目中。
- singleton:單例模式,當spring 創建applicationContext 容器的時候,spring會預初始化所有的該作用域實例,加上lazy-init 就可以避免預處理;
- prototype:原型模式,每次通過getBean 獲取該bean 就會新產生一個實例,創建后spring 將不再對其管理;
- request:每次請求都新產生一個實例,和prototype 不同就是創建后,接下來的管理,spring依然在監聽;
- session:每次會話,同上;
- global session:全局的web 域,類似于servlet 中的application。
到此這篇關于淺談Spring 的Controller 是單例or多例的文章就介紹到這了,更多相關Spring Controller 單例or多例內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!
原文鏈接:https://www.cnblogs.com/taojietaoge/p/13557447.html