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