自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

你會(huì)用Java代碼模擬高并發(fā)嗎?

開發(fā) 后端
Java通過(guò)代碼模擬高并發(fā)可以以最快的方式發(fā)現(xiàn)我們系統(tǒng)中潛在的線程安全性問(wèn)題,此處使用Semaphore(信號(hào)量)和 CountDownLatch(閉鎖)搭配ExecutorService(線程池)來(lái)進(jìn)行模擬。

 [[271769]]

 

 

Java通過(guò)代碼模擬高并發(fā)可以以最快的方式發(fā)現(xiàn)我們系統(tǒng)中潛在的線程安全性問(wèn)題,此處使用Semaphore(信號(hào)量)和 CountDownLatch(閉鎖)搭配ExecutorService(線程池)來(lái)進(jìn)行模擬,主要介紹如下:

1、Semaphore

JDK 1.5之后會(huì)提供這個(gè)類

Semaphore是一種基于計(jì)數(shù)的信號(hào)量。它可以設(shè)定一個(gè)閾值,基于此,多個(gè)線程競(jìng)爭(zhēng)獲取許可信號(hào),做完自己的申請(qǐng)后歸還,超過(guò)閾值后,線程申請(qǐng)?jiān)S可信號(hào)將會(huì)被阻塞。Semaphore可以用來(lái)構(gòu)建一些對(duì)象池,資源池之類的,比如數(shù)據(jù)庫(kù)連接池,我們也可以創(chuàng)建計(jì)數(shù)為1的Semaphore,將其作為一種類似互斥鎖的機(jī)制,這也叫二元信號(hào)量,表示兩種互斥狀態(tài)。

2、CountDownLatch

JDK 1.5之后會(huì)提供這個(gè)類,

CountDownLatch這個(gè)類能夠使一個(gè)線程等待其他線程完成各自的工作后再執(zhí)行。例如,應(yīng)用程序的主線程希望在負(fù)責(zé)啟動(dòng)框架服務(wù)的線程已經(jīng)啟動(dòng)所有的框架服務(wù)之后再執(zhí)行。

CountDownLatch是通過(guò)一個(gè)計(jì)數(shù)器來(lái)實(shí)現(xiàn)的,計(jì)數(shù)器的初始值為線程的數(shù)量。每當(dāng)一個(gè)線程完成了自己的任務(wù)后,計(jì)數(shù)器的值就會(huì)減1。當(dāng)計(jì)數(shù)器值到達(dá)0時(shí),它表示所有的線程已經(jīng)完成了任務(wù),然后在閉鎖上等待的線程就可以恢復(fù)執(zhí)行任務(wù)。

如下圖:

你會(huì)用Java代碼模擬高并發(fā)嗎?

 

以上兩個(gè)類可以搭配使用,達(dá)到模擬高并發(fā)的效果,以下使用代碼的形式進(jìn)行舉例:

  1. package modules; 
  2. import java.util.concurrent.CountDownLatch; 
  3. import java.util.concurrent.ExecutorService; 
  4. import java.util.concurrent.Executors; 
  5. import java.util.concurrent.Semaphore; 
  6. public class CountExample { 
  7.  // 請(qǐng)求總數(shù) 
  8.  public static int clientTotal = 5000; 
  9.  // 同時(shí)并發(fā)執(zhí)行的線程數(shù) 
  10.  public static int threadTotal = 200; 
  11.  public static int count = 0; 
  12.  public static void main(String[] args) throws Exception { 
  13.  ExecutorService executorService = Executors.newCachedThreadPool(); 
  14.  //信號(hào)量,此處用于控制并發(fā)的線程數(shù) 
  15.  final Semaphore semaphore = new Semaphore(threadTotal); 
  16.  //閉鎖,可實(shí)現(xiàn)計(jì)數(shù)器遞減 
  17.  final CountDownLatch countDownLatch = new CountDownLatch(clientTotal); 
  18.  for (int i = 0; i < clientTotal ; i++) { 
  19.  executorService.execute(() -> { 
  20.  try { 
  21.     //執(zhí)行此方法用于獲取執(zhí)行許可,當(dāng)總計(jì)未釋放的許可數(shù)不超過(guò)200時(shí), 
  22.     //允許通行,否則線程阻塞等待,直到獲取到許可。 
  23.  semaphore.acquire(); 
  24.  add(); 
  25.  //釋放許可 
  26.  semaphore.release(); 
  27.  } catch (Exception e) { 
  28.  //log.error("exception", e); 
  29.  e.printStackTrace(); 
  30.  } 
  31.  //閉鎖減一 
  32.  countDownLatch.countDown(); 
  33.  }); 
  34.  } 
  35.  countDownLatch.await();//線程阻塞,直到閉鎖值為0時(shí),阻塞才釋放,繼續(xù)往下執(zhí)行 
  36.  executorService.shutdown(); 
  37.  log.info("count:{}"count); 
  38.  } 
  39.  private static void add() { 
  40.  count++; 
  41.  } 

如上方法模擬5000次請(qǐng)求,同時(shí)最大200個(gè)并發(fā)操作,觀察最后的結(jié)果,發(fā)現(xiàn)每次的結(jié)果都有差別,和預(yù)期不符,得出結(jié)果部分如下:

  1. 22:18:26.449 [main] INFO modules.CountExample - count:4997 
  2. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  3. 22:18:26.449 [main] INFO modules.CountExample - count:4995 
  4. 22:18:26.449 [main] INFO modules.CountExample - count:4998 

最后結(jié)論:add 方法 非線程安全

那如何保證add方法 線程安全,將add方法進(jìn)行如下修改即可:

  1. private static void add() { 
  2.  count.incrementAndGet(); 

執(zhí)行結(jié)果如下:

  1. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  2. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  3. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  4. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  5. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  6. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  7. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  8. 22:18:26.449 [main] INFO modules.CountExample - count:5000 

最后結(jié)論:修改后 的 add 方法 線程安全

責(zé)任編輯:龐桂玉 來(lái)源: 今日頭條
相關(guān)推薦

2025-01-20 00:00:00

反射Java語(yǔ)言

2018-09-29 15:34:34

JavaList接口

2020-06-04 14:15:55

Java中BigDecimal函數(shù)

2021-08-11 10:00:51

緩存MyBatis管理

2021-09-06 10:42:18

Linux命令服務(wù)器

2021-05-21 12:36:16

限流代碼Java

2024-03-06 08:15:03

@Autowired注入方式Spring

2021-01-28 09:55:50

while(1)for(;;)Linux

2021-03-17 11:16:58

while(1)for(;;)語(yǔ)言

2019-01-28 17:42:33

Python數(shù)據(jù)預(yù)處理數(shù)據(jù)標(biāo)準(zhǔn)化

2023-12-01 11:13:50

JavaTreeSet

2023-12-07 08:13:58

Java開發(fā)

2017-02-20 07:47:04

緩存HASH高并發(fā)

2020-11-09 09:03:35

高并發(fā)多線程ThreadLocal

2020-06-29 08:32:21

高并發(fā)程序員流量

2022-02-10 09:04:50

架構(gòu)

2024-04-08 00:00:00

asyncawaiPromise

2020-12-28 08:25:08

Java并發(fā)HashMap

2018-12-20 09:30:59

分布式高并發(fā)多線程

2022-11-07 17:50:36

點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)