Java使用程式碼模擬高併發操作
摘要: 在java中,使用了synchronized關鍵字和Lock鎖實現了資源的併發訪問控制,在同一時間只允許唯一了執行緒進入臨界區訪問資源(讀鎖除外),這樣子控制的主要目的是為了解決多個執行緒併發同一資源造成的資料不一致的問題。在另外一種場景下,一個資源有多個副本可供同時使用,比如印表機房有多個印表機、廁所有多個坑可供同時使用,這種情況下,Java提供了另外的併發訪問控制--資源的多副本的併發訪問控制,今天使用的Semaphore即是其中的一種。
Java透過程式碼模擬高併發可以以最快的方式發現我們系統中潛在的執行緒安全性問題,此處使用Semaphore(訊號量)和 CountDownLatch(閉鎖)搭配ExecutorService(執行緒池)來進行模擬,主要介紹如下:
1、Semaphore
JDK 1.5之後會提供這個類
Semaphore是一種基於計數的訊號量。它可以設定一個閾值,基於此,多個執行緒競爭獲取許可訊號,做完自己的申請後歸還,超過閾值後,執行緒申請許可訊號將會被阻塞。Semaphore可以用來構建一些物件池,資源池之類的,比如資料庫連線池,我們也可以建立計數為1的Semaphore,將其作為一種類似互斥鎖的機制,這也叫二元訊號量,表示兩種互斥狀態。
2、CountDownLatch
JDK 1.5之後會提供這個類,
CountDownLatch這個類能夠使一個執行緒等待其他執行緒完成各自的工作後再執行。例如,應用程式的主執行緒希望在負責啟動框架服務的執行緒已經啟動所有的框架服務之後再執行。
CountDownLatch是透過一個計數器來實現的,計數器的初始值為執行緒的數量。每當一個執行緒完成了自己的任務後,計數器的值就會減1。當計數器值到達0時,它表示所有的執行緒已經完成了任務,然後在閉鎖上等待的執行緒就可以恢復執行任務。
如下圖:
以上兩個類可以搭配使用,達到模擬高併發的效果,以下使用程式碼的形式進行舉例:
package modules;import java.util.concurrent.CountDownLatch;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;import java.util.concurrent.Semaphore;public class CountExample { // 請求總數 public static int clientTotal = 5000; // 同時併發執行的執行緒數 public static int threadTotal = 200; public static int count = 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 { try { //執行此方法用於獲取執行許可,當總計未釋放的許可數不超過200時, //允許通行,否則執行緒阻塞等待,直到獲取到許可。 semaphore.acquire(); add(); //釋放許可 semaphore.release(); } catch (Exception e) { //log.error("exception", e); e.printStackTrace(); } //閉鎖減一 countDownLatch.countDown(); }); } countDownLatch.await();//執行緒阻塞,直到閉鎖值為0時,阻塞才釋放,繼續往下執行 executorService.shutdown(); log.info("count:{}", count); } private static void add() { count++; } }
如上方法模擬5000次請求,同時最大200個併發操作,觀察最後的結果,發現每次的結果都有差別,和預期不符,得出結果部分如下:
22:18:26.449 [main] INFO modules.CountExample - count:499722:18:26.449 [main] INFO modules.CountExample - count:500022:18:26.449 [main] INFO modules.CountExample - count:499522:18:26.449 [main] INFO modules.CountExample - count:4998
最後結論:add 方法 非執行緒安全
那如何保證add方法 執行緒安全,將add方法進行如下修改即可:
private static void add() { count.incrementAndGet(); }
執行結果如下:
22:18:26.449 [main] INFO modules.CountExample - count:500022:18:26.449 [main] INFO modules.CountExample - count:500022:18:26.449 [main] INFO modules.CountExample - count:500022:18:26.449 [main] INFO modules.CountExample - count:500022:18:26.449 [main] INFO modules.CountExample - count:500022:18:26.449 [main] INFO modules.CountExample - count:500022:18:26.449 [main] INFO modules.CountExample - count:500022:18:26.449 [main] INFO modules.CountExample - count:5000
最後結論:修改後 的 add 方法 執行緒安全
作者:Oo若離oO
來源:https://my.oschina.net/ruoli/blog/1801866
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/1806/viewspace-2809002/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- 併發模擬-程式碼CountDownLatch,SemaphoreCountDownLatch
- Laravel 高併發搶購模擬Laravel
- Redis 快取穿透,擊穿解決方案,模擬高併發請求,附程式碼Redis快取穿透
- java:模擬棧操作Java
- java高併發系列 - 第21天:java中的CAS操作,java併發的基石Java
- Java併發程式設計之Java CAS操作Java程式設計
- 使用java程式碼操作redisJavaRedis
- Java 高併發思路Java
- 模擬併發及生成唯一
- Java 併發程式設計:ThreadLocal 的使用及其原始碼實現Java程式設計thread原始碼
- 《java併發程式設計的藝術》原子操作類Java程式設計
- 實戰Java高併發程式設計模式視訊Java程式設計設計模式
- 走進高併發(二)Java並行程式基礎Java並行行程
- 高併發mysql update操作必定失敗MySql
- java 併發程式設計-AQS原始碼分析Java程式設計AQS原始碼
- 程式碼審查清單:Java併發 - Roman LeventovJava
- Java併發程式設計非同步操作Future和FutureTaskJava程式設計非同步
- java併發程式設計:Thread類的使用Java程式設計thread
- Java 併發包原子操作類解析Java
- 2024.9.28 程式碼源模擬賽
- (四)Java高併發秒殺API之高併發優化JavaAPI優化
- Java 併發程式設計(十五) -- Semaphore原始碼分析Java程式設計原始碼
- Java 併發程式設計(十四) -- CyclicBarrier原始碼分析Java程式設計原始碼
- Java 併發程式設計(十三) -- CountDownLatch原始碼分析Java程式設計CountDownLatch原始碼
- Java 併發程式設計(九) -- ReentrantLock 原始碼分析Java程式設計ReentrantLock原始碼
- Java 併發程式設計(七) -- AbstractQueuedSynchronizer 原始碼分析Java程式設計原始碼
- Java高併發之CyclicBarrier簡介Java
- Java高併發測試框架JCStressJava框架
- java併發程式設計系列:java併發程式設計背景知識Java程式設計
- Java併發程式設計之CyclicBarrier使用指南Java程式設計
- 前端模擬使用者的複製操作前端
- java 併發程式設計Java程式設計
- Java併發程式設計Java程式設計
- 【Java程式設計】使用Java模擬C/C++中的queue佇列Java程式設計C++佇列
- Java併發程式設計-鎖及併發容器Java程式設計
- 高併發網路程式設計程式設計
- 使用Netty模擬發生OOMNettyOOM
- Springboot:高併發下耗時操作的實現Spring Boot