一:servlet線程安全問題發生的條件
如果多個客戶端訪問同一個servlet時,發生線程安全問題,那么它們訪問的是相同的資源。如果訪問
的不是相同資源,則不存在線程安全問題。
實例1:不會產生線程安全問題,因為每個客戶端發送請求,都會創建一個線程,都會創建一個count
不存在資源共享的問題。
1 public void doPost(HttpServletRequest request, HttpServletResponse response)
2 throws ServletException, IOException { 3 int count = 0; 4 count++; 5 response.getOutputStream().write((count + "").getBytes()); 6 }
實例2:這種方式,多個線程公用資源,應該存在線程安全問題,但是我的測試結果一直不存在線程安全問題。有點不解?
1 public class ServletDemo extends HttpServlet {
2
3 int count = 0; 4 5 public void doGet(HttpServletRequest request, HttpServletResponse response) 6 throws ServletException, IOException { 7 count++; 8 try { 9 Thread.sleep(1000 * 10); 10 } catch (InterruptedException e) { 11 e.printStackTrace(); 12 } 13 response.getOutputStream().write((count + "").getBytes()); 14 } 15 16 }
二:線程安全問題的處理
1:對於線程安全問題最簡單的方式就是加鎖:
將存在線程安全問題的代碼放到同步代碼塊中,這樣線程訪問時就需要排隊拿到鑰匙,只有上一個
線程訪問完畢,才會釋放掉鎖,先一個線程才可以進入。但是存在明顯的缺點:就是效率太低了。
例如:門戶網站日訪問量過千萬,效率太低了。
注意:應該盡量減少代碼在同步代碼塊中
1 synchronized (this) {
2 count++; 3 try { 4 Thread.sleep(1000 * 10); 5 } catch (InterruptedException e) { 6 e.printStackTrace(); 7 } 8 response.getOutputStream().write((count + "").getBytes()); 9 }
2:實現singleThreadModel接口
servlet實現singleThreadModel接口后,每個線程都會創建servlet實例,這樣每個客戶端請求就不存在共享資源的問題
,但是servlet響應客戶端請求的效率太低,所以已經淘汰。