面試常問問題:銀行網上支付項目中怎麼控制多線程高併發訪問?

面試常問問題:銀行網上支付項目中怎麼控制多線程高併發訪問?

 

synchronized關鍵字主要解決多線程共享數據同步問題。
ThreadLocal使用場合主要解決多線程中數據因併發產生不一致問題。
java

ThreadLocal和Synchonized都用於解決多線程併發訪問。可是ThreadLocal與synchronized有本質的區別:
    synchronized是利用鎖的機制,使變量或代碼塊在某一時該只能被一個線程訪問。而ThreadLocal爲每個線程都提供了變量的副本,使 得每一個線程在某一時間訪問到的並非同一個對象,這樣就隔離了多個線程對數據的數據共享。而Synchronized卻正好相反,它用於在多個線程間通訊 時可以得到數據共享。
程序員

Synchronized用於線程間的數據共享,而ThreadLocal則用於線程間的數據隔離。固然ThreadLocal並不能替代synchronized,它們處理不一樣的問題域。Synchronized用於實現同步機制,比ThreadLocal更加複雜。面試

一、java中synchronized用法
    使用了synchronized關鍵字能夠輕鬆地解決多線程共享數據同步問題。
    synchronized關鍵字能夠做爲函數的修飾符,也可做爲函數內的語句,也就是平時說的同步方法和同步語句塊。若是再細的分 類,synchronized可做用於instance變量、object reference(對象引用)、static函數和class literals(類名稱字面常量)身上。  
       synchronized取得的鎖都是對象;每一個對象只有一個鎖(lock)與之相關聯;實現同步是要很大的系統開銷做爲代價的,甚至可能形成死鎖,因此儘可能避免無謂的同步控制。
多線程

synchronized的4種用法 :
    1.方法聲明時使用,線程得到的是成員鎖.
    2.對某一代碼塊使用,synchronized後跟括號,括號裏是變量,線程得到的是成員鎖.
    3.synchronized後面括號裏是一對象,此時,線程得到的是對象鎖.
    4.synchronized後面括號裏是類,此時,線程得到的是對象鎖.
併發

二、java.lang.ThreadLocal()的用法  
1、概述
ThreadLocal 是什麼呢?其實ThreadLocal並不是是一個線程的本地實現版本,它並非一個Thread,而是 threadlocalvariable(線程局部變量)。也許把它命名爲ThreadLocalVar更加合適。線程局部變量 (ThreadLocal)其實的功用很是簡單,就是爲每個使用該變量的線程都提供一個變量值的副本,是Java中一種較爲特殊的線程綁定機制,是每一 個線程均可以獨立地改變本身的副本,而不會和其它線程的副本衝突。
函數

ThreadLocal是如何作到爲每個線程維護變量的副本的呢?其實實現的思路很簡單,在ThreadLocal類中有一個Map,用於存儲每個線程的變量的副本。高併發

歸納起來講,對於多線程資源共享的問題,同步機制採用了「以時間換空間」的方式,而ThreadLocal採用了「以空間換時間」的方式。前者僅提供一份變量,讓不一樣的線程排隊訪問,然後者爲每個線程都提供了一份變量,所以能夠同時訪問而互不影響。性能

2、API說明spa

ThreadLocal()
    建立一個線程本地變量。
線程

T get()
    返回此線程局部變量的當前線程副本中的值,若是這是線程第一次調用該方法,則建立並初始化此副本。

protected T initialValue()
    返回此線程局部變量的當前線程的初始值。最多在每次訪問線程來得到每一個線程局部變量時調用此方法一次,即線程第一次使用 get() 方法訪問變量的時候。若是線程先於 get 方法調用 set(T) 方法,則不會在線程中再調用 initialValue 方法。
    若該實現只返回 null;若是程序員但願將線程局部變量初始化爲 null 之外的某個值,則必須爲 ThreadLocal 建立子類,並重寫此方法。一般,將使用匿名內部類。initialValue 的典型實現將調用一個適當的構造方法,並返回新構造的對象。

void remove()
    移除此線程局部變量的值。這可能有助於減小線程局部變量的存儲需求。若是再次訪問此線程局部變量,那麼在默認狀況下它將擁有其 initialValue。

void set(T value)
    將此線程局部變量的當前線程副本中的值設置爲指定值。許多應用程序不須要這項功能,它們只依賴於 initialValue() 方法來設置線程局部變量的值。
    在程序中通常都重寫initialValue方法,以給定一個特定的初始值。

3、典型實例

4、總結
    ThreadLocal使用場合主要解決多線程中數據因併發產生不一致問題。
    
    ThreadLocal爲每一個線程的中併發訪問的數據提供一個副本,經過訪問副原本運行業務,這樣的結果是耗費了內存,單大大減小了線程同步所帶來性能消耗,也減小了線程併發控制的複雜度。

    ThreadLocal不能使用原子類型,只能使用Object類型。ThreadLocal的使用比synchronized要簡單得多。

5、ThreadLocal使用的通常步驟     一、在多線程的類(如ThreadDemo類)中,建立一個ThreadLocal對象threadXxx,用來保存線程間須要隔離處理的對象xxx。     二、在ThreadDemo類中,建立一個獲取要隔離訪問的數據的方法getXxx(),在方法中判斷,若ThreadLocal對象爲null時候,應該new()一個隔離訪問類型的對象,並強制轉換爲要應用的類型。     三、在ThreadDemo類的run()方法中,經過getXxx()方法獲取要操做的數據,這樣能夠保證每一個線程對應一個數據對象,在任什麼時候刻都操做的是這個對象。