摘要:線程安全性原子性提供了互斥訪問,同一時刻只能有一個線程來對他進行操作。原子性包是通過來保證線程原子性通過比較操作的對象的值工作內存的值與底層的值共享內存中的值對比是否相同來判斷是否進行處理,如果不相同則重新獲取。
線程安全性 定義
當多個線程訪問同一個類時,不管運行時環境采用何種調度方式,不論線程如何交替執行,在主調代碼中不需要額外的協同或者同步代碼時,這個類都可以表現出正確的行為,我們則稱這個類為線程安全的。線程安全性
原子性:提供了互斥訪問,同一時刻只能有一個線程來對他進行操作。
可見性:一個線程對主內存的修改可以及時被其他線程觀察到。
有序性:一個線程觀察其他線程中的指令順序,由于指令重排序的存在,該結果一般雜亂無序。
原子性 - Atomic包AtomicXXX 是通過 CAS(CompareAndSwap)來保證線程原子性 通過比較操作的對象的值(工作內存的值)與底層的值(共享內存中的值)對比是否相同來判斷是否進行處理,如果不相同則重新獲取。如此循環操作,直至獲取到期望的值。
(關于什么是主內存什么事工作內存在上篇博客中進行介紹了,不懂的同學可以翻一下)示例代碼:
@Slf4j public class AtomicExample2 { // 請求總數 public static int clientTotal = 5000; // 同時并發執行的線程數 public static int threadTotal = 200; public static AtomicLong count = new AtomicLong(0); public static void main(String[] args) throws Exception { ExecutorService executorService = Executors.newCachedThreadPool(); final Semaphore semaphore = new Semaphore(threadTotal); final CountDownLatch countDownLatch = new CountDownLatch(clientTotal); for (int i = 0; i < clientTotal ; i++) { executorService.execute(() -> { try { semaphore.acquire(); add(); semaphore.release(); } catch (Exception e) { log.error("exception", e); } countDownLatch.countDown(); }); } countDownLatch.await(); executorService.shutdown(); log.info("count:{}", count.get()); } private static void add() { count.incrementAndGet(); // count.getAndIncrement(); } }
LongAdder和DoubleAdder
jdk8中新增的保證同步操作的類,我們之前介紹了AtomicXXX來保證原子性,那么為什么還有有LongAdder呢?
說AtomicXXX的實現是通過死循環來判斷值的,在低并發的情況下AtomicXXX進行更改值的命中率還是很高的。但是在高并發下進行命中率可能沒有那么高,從而一直執行循環操作,此時存在一定的性能消耗,在jvm中我們允許將64位的數值拆分成2個32位的數進行儲存的,LongAdder的思想就是將熱點數據分離,將AtomicXXX中的核心數據分離,熱點數據會被分離成多個數組,每個數據都多帶帶維護各自的值,將單點的并行壓力發散到了各個節點,這樣就提高了并行,在低并發的時候性能基本和AtomicXXX相同,在高并發時具有較好的性能,缺點是在并發更新時統計時可能會出現誤差。在低并發,需要全局唯一,準確的比如id等使用AtomicXXX,要求性能使用LongAdder
@Slf4j public class AtomicExample3 { // 請求總數 public static int clientTotal = 5000; // 同時并發執行的線程數 public static int threadTotal = 200; public static LongAdder count = new LongAdder(); public static void main(String[] args) throws Exception { ExecutorService executorService = Executors.newCachedThreadPool(); final Semaphore semaphore = new Semaphore(threadTotal); final CountDownLatch countDownLatch = new CountDownLatch(clientTotal);、】【poiuytrewq;" for (int i = 0; i < clientTotal ; i++) { executorService.execute(() -> { try { semaphore.acquire(); add(); semaphore.release(); } catch (Exception e) { log.error("exception", e); } countDownLatch.countDown(); }); } countDownLatch.await(); executorService.shutdown(); log.info("count:{}", count); } private static void add() { count.increment(); } }
AtomicReference、AtomicReferenceFieldUpdater
AtomicReference是給定指定的期望值當期望值與主內存中的值相同然后更新,示例代碼
@Slf4j public class AtomicExample4 { private static AtomicReferencecount = new AtomicReference<>(0); public static void main(String[] args) { count.compareAndSet(0, 2); // 2 count.compareAndSet(0, 1); // no count.compareAndSet(1, 3); // no count.compareAndSet(2, 4); // 4 count.compareAndSet(3, 5); // no log.info("count:{}", count.get()); } }
AtomMNBVCXZenceFieldUpdater主要是更新某一個實例對象的一個字段這個字段必須是用volatile修飾同時不能是private修飾的,·157-=· 123444457890-
@Slf4j public class AtomicExample5 { private static AtomicIntegerFieldUpdaterupdater = AtomicIntegerFieldUpdater.newUpdater(AtomicExample5.class, "count"); @Getter public volatile int count = 100; public static void main(String[] args) { AtomicExample5 example5 = new AtomicExample5(); if (updater.compareAndSet(example5, 100, 120)) { log.info("update success 1, {}", example5.getCount()); } if (updater.compareAndSet(example5, 100, 120)) { log.info("update success 2, {}", example5.getCount()); } else { log.info("update failed, {}", example5.getCount()); } } }
最后我們介紹一下使用AtomicBoolean來實現只執行一次的操作,我們使用private static AtomicBoolean isHappened = new AtomicBoolean(false)來初始化一個具有原子性的一個Boolean的記錄是否已經被執行
@Slf4j public class AtomicExample6 { private static AtomicBoolean isHappened = new AtomicBoolean(false); // 請求總數 public static int clientTotal = 5000; // 同時并發執行的線程數 public static int threadTotal = 200; public static void main(String[] args) throws Exception { ExecutorService executorService = Executors.newCachedThreadPool(); final Semaphore semaphore = new Semaphore(threadTotal); final CountDownLatch countDownLatch = new CountDownLatch(clientTotal); for (int i = 0; i < clientTotal ; i++) { executorService.execute(() -> { try { semaphore.acquire(); test(); semaphore.release(); } catch (Exception e) { log.error("exception", e); } countDownLatch.countDown(); }); } countDownLatch.await(); executorService.shutdown(); log.info("isHappened:{}", isHappened.get()); } private static void test() { if (isHappened.compareAndSet(false, true)) { log.info("execute"); } } }原子性 - 鎖
我們除了可以使用Atomic包還可以使用鎖來實現。
synchronize:依賴jvm
修飾代碼塊:適用范圍大括號括起來的代碼,作用于調用的對象
修飾方法:適用范圍整個方法,作用于調用的對象
修飾靜態方法:適用范圍整個靜態方法,作用于所有對象
修飾一個類:適用范圍是括起來的部分,作用于所有對象
Lock:依賴特殊的cpu指令、代碼實現,ReentrantLock
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/74433.html
摘要:是需要我們去處理很多事情,為了防止多線程給我們帶來的安全和性能的問題下面就來簡單總結一下我們需要哪些知識點來解決多線程遇到的問題。 前言 不小心就鴿了幾天沒有更新了,這個星期回家咯。在學校的日子要努力一點才行! 只有光頭才能變強 回顧前面: 多線程三分鐘就可以入個門了! Thread源碼剖析 本文章的知識主要參考《Java并發編程實戰》這本書的前4章,這本書的前4章都是講解并發的基...
摘要:將與當前線程建立一對一關系的值移除。為了讓方法里的操作具有原子性,也就是在一個線程執行這一系列操作的同時禁止其他線程執行這些操作,提出了鎖的概念。 上頭一直在說以線程為基礎的并發編程的好處了,什么提高處理器利用率啦,簡化編程模型啦。但是磚家們還是認為并發編程是程序開發中最不可捉摸、最詭異、最扯犢子、最麻煩、最惡心、最心煩、最容易出錯、最不符合社會主義核心價值觀的一個部分~ 造成這么多最...
摘要:另一個是使用鎖的機制來處理線程之間的原子性。依賴于去實現鎖,因此在這個關鍵字作用對象的作用范圍內,都是同一時刻只能有一個線程對其進行操作的。 線程安全性 定義:當多個線程訪問某個類時,不管運行時環境采用何種調度方式或者這些線程將如何交替執行,并且在主調代碼中不需要任何額外的同步或協同,這個類都能表現出正確的行為,那么就稱這個類是線程安全的。 線程安全性主要體現在三個方面:原子性、可見性...
摘要:線程的這種交叉操作會導致線程不安全。原子操作是在多線程環境下避免數據不一致必須的手段。如果聲明一個域為一些情況就可以確保多線程訪問到的變量是最新的。并發要求一個線程對對象進行了操作,對象發生了變化,這種變化應該對其他線程是可見的。 雖是讀書筆記,但是如轉載請注明出處 http://segmentfault.com/blog/exploring/ .. 拒絕伸手復制黨 一個問題: ...
摘要:的線程機制是搶占式。會讓出當多個線程并發的對主存中的數據進行操作時,有且只有一個會成功,其余均失敗。和對象只有在困難的多線程問題中才是必須的。 并發簡述 并發通常是用于提高運行在單處理器上的程序的性能。在單 CPU 機器上使用多任務的程序在任意時刻只在執行一項工作。 并發編程使得一個程序可以被劃分為多個分離的、獨立的任務。一個線程就是在進程中的一個單一的順序控制流。java的線程機制是...
閱讀 604·2021-11-18 13:12
閱讀 1321·2021-11-15 11:39
閱讀 2480·2021-09-23 11:22
閱讀 6212·2021-09-22 15:15
閱讀 3665·2021-09-02 09:54
閱讀 2318·2019-08-30 11:10
閱讀 3250·2019-08-29 14:13
閱讀 2918·2019-08-29 12:49