Java實現線程安全的方式
多線程環境中如何保證線程安全?java可以實現線程安全的方式歸納如下:
1、使用synchronized關鍵字
synchronized關鍵字可以修飾方法和代碼塊,它的語義是保證同一段代碼同一時間只能有一個線程在執行。
2、使用volatile關鍵字
volatile關鍵字用來修飾共享變量。它的語義是保證被修飾的變量在被一個線程修改后,都會通知其他線程,其他線程需要操作該變量時會重新獲取,這樣每個線程在操作該共享變量時獲取到的值都是很新的,但是volatile關鍵字無法保證原子性,怎么來理解這句話呢?在你寫的對該變量操作的一行代碼,在執行時是拆分成多條字節碼指令后執行的,所以對volatile修飾的變量操作時,需要保證該操作具有原子性。有些操作本身就具有原子性,無需額外編碼,比如:volatile修飾boolean類型的變量,對該變量的操作是賦值操作。對于不具備原子性的操作,則需要使用某些方法來保證原子性,比如:使用原子類來替代基本數據類型
3、使用原子類代替基本數據類型
java提供三種類型的原子類,當某個操作因為不是原子操作導致的線程安全問題的時候,可以使用原子類來替代。比如:多線程環境下執行a++,可以使用AtomicInteger類incrementAndGet()方法實現。相比synchronized,原子類是使用樂觀鎖來實現線程安全,synchronized使用悲觀鎖來實現線程安全。
4、使用ThreadLocal對各個線程進行隔離
使用ThreadLocal保存當前線程的變量值,這樣你想獲取該變量的值的時候,獲取到的都是本線程的變量值,不會獲取到其他線程設置的值。應用場景比如:在過濾器中使用threadlocal保存變量值,需要使用的時候直接取出來即可,threadlocal可以保證你取到的數據就是你之前設置的,因為在一個線程中。