Java并發編程之由于靜態變量錯誤使用可能導致的并發問題

Java并發編程之由于靜態變量錯誤使用可能導致的并發問題

    • 1.1 前言
    • 1.2 業務背景
    • 1.3 問題分析
    • 1.4 為什么呢?
    • 1.5 修復方案
    • 2 演示示例源碼下載

1.1 前言

我們知道在 Java 后端服務開發中,如果出現并發問題一般都是由于在多個線程中使用了共享的變量導致的。

今天我們就一起來看一個由于靜態變量錯誤使用可能導致的并發問題。

PS:

  • 今天分享的這個案例,只有程序啟動后,首次調用的時候就開并發調用才可能出現,所以相對比較隱蔽。
  • 由于這個并發問題不是百分百出現,所以差點埋下雷,幸好在 QA發現后,在不相信玄學的組長的領導下,終于發現了罪魁禍首。
  • 當時發現問題的表象就是:同樣的代碼同樣的鏡像,服務部署完成后是有問題的,重啟后可能就沒問題了。
  • 以下代碼示例均已脫敏,簡化了非問題相關部分。

1.2 業務背景

為了優化程序的執行性能,避免每次創建都初始化一個集合,因此創建了一個靜態工具類ConstantUtils.java

import java.util.HashSet;
import java.util.Set;public class ConstantUtils {private static Set<String> goodApiList=new HashSet<>();public static Set<String> getGoodApiList() {if(goodApiList.isEmpty()){goodApiList.add("A");goodApiList.add("B");goodApiList.add("C");goodApiList.add("D");goodApiList.add("E");goodApiList.add("F");goodApiList.add("G");goodApiList.add("H");goodApiList.add("I");goodApiList.add("J");goodApiList.add("K");}return goodApiList;}/*** 靜態工具類應該禁用其構造方法*/private ConstantUtils(){}
}

最開始的思路是嘗試通過懶加載在靜態方法中初始化了一些值,這樣只有調用ConstantUtils.getGoodApiList()方法的時候才會初始化。

然后接下來,如果并發調用接口則會并發執行calculate 這個方法,也就是說會并發執行new MyCounter();

public class MyProscessor{public void calculate(Map<String, String> dbResult) throws Exception {MyProxy myProxy = new MyProxy(); myProxy.setCounter(new MyCounter());return myProxy.doCount(dbResult);}
}

然后我們在這個MyCounter的成員變量中調用了ConstantUtils.getGoodApiList();方法,避免在new出來的不同的MyCounter實例中多次創建 goodApiList集合。

public class MyCounter(){private final Set<String> goodApiList= ConstantUtils.getGoodApiList();public void doCount(Map<String, String> dbResult){Long count=0L;if(goodApiList.contains("xxxx")){count++;....}...}
}

最后判斷做了統計數量。

可以確定的是 dbResult 中的結果是固定不變的,但是同樣的代碼,同樣的鏡像,部署兩次后,卻統計出來的 count 數量卻不一致。

而且經過最終排查后發現 goodApiList 的結果可能出現多種情況,比如下面兩種情況:

情況一:HashSet 集合里面確實 A,B,C,D,E,F,G,H,I,J,K ,但是調用size方法不是 11 而是 13 甚至 15
情況二:HashSet 集合里面確實 A,B,C,D,E, ,但是調用size方法不是 11 而是 13 甚至 16

看到這里,你看到問題出在哪里了么?

1.3 問題分析

如果你看出來了,那么恭喜你,很機智。

如果沒看出來,也沒關系,我們一起梳理下思路:

  1. 首先我們在一個類中,如果并發調用接口導致并發執行了new MyCounter();
  2. 然后在 new MyCounter();的實例中調用了ConstantUtils.getGoodApiList();
  3. 然而 getGoodApiList()方法是這樣實現的:
public static Set<String> getGoodApiList() {if(goodApiList.isEmpty()){goodApiList.add("A");goodApiList.add("B");goodApiList.add("C");goodApiList.add("D");goodApiList.add("E");goodApiList.add("F");goodApiList.add("G");goodApiList.add("H");goodApiList.add("I");goodApiList.add("J");goodApiList.add("K");}return goodApiList;
}

上面代碼執行可能出現這樣一個情況,當程序剛部署玩后,假設一個線程A執行到添加 B之后,兩外一個線程也進來了。

public static Set<String> getGoodApiList() {if(goodApiList.isEmpty()){goodApiList.add("A");goodApiList.add("B");// ... 假設線程 A 此時執行到這里了goodApiList.add("C");goodApiList.add("D");goodApiList.add("E");goodApiList.add("F");goodApiList.add("G");goodApiList.add("H");goodApiList.add("I");goodApiList.add("J");goodApiList.add("K");}return goodApiList;
}

線程 B 執行的時候,由于線程 A已經給goodApiList放入了幾個值了,因此此時goodApiList.isEmpty 為 false,直接返回了 goodApiList.

那等十分鐘后再次非并發調用呢?結果會恢復正常么?

按照最開始的想法,不管怎么樣,最終線程 A 執行完,

那么goodApiList里面放的元素肯定就是 A,B,C,D,E,F,G,H,I,J,K 且 goodApiList.size() 就是 11了,是么?

如果你也這么認為那就大錯特錯了。

為了測試,我們這里用一個測試類來復現這個問題

import junit.framework.TestCase;import java.util.Set;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;/**** @author qingfeng.zhao* @date 2024/6/6* @apiNote*/
public class ConstantUtilsTest extends TestCase {private static final int THREAD_NUM = 100; // 模擬100個并發線程private static final int COUNT_DOWN = THREAD_NUM; // 計數器public void setUp() throws Exception {super.setUp();}public void tearDown() throws Exception {}public void testGetGoodApiList()throws InterruptedException {ExecutorService executorService = Executors.newFixedThreadPool(THREAD_NUM);CountDownLatch latch = new CountDownLatch(COUNT_DOWN);for (int i = 0; i < THREAD_NUM; i++) {executorService.submit(() -> {try {// 模擬業務邏輯,調用ConstantUtils.getGoodsList()方法Set<String> goodList= ConstantUtils.getGoodApiList();} finally {latch.countDown(); // 線程執行完畢,計數器減一}});}latch.await(); // 等待所有線程執行完畢executorService.shutdown(); // 關閉線程池System.out.println("所有線程執行完畢!");for (int i = 0; i < 10; i++) {Set<String> goodList=ConstantUtils.getGoodApiList();System.out.println("-----start--------------");for (String item:goodList){System.out.println(item+"---"+goodList.size());}System.out.println("-----end--------------");}}
}

程序執行有時候會是這個樣子:
在這里插入圖片描述
有時候又會是這個樣子:
在這里插入圖片描述

1.4 為什么呢?

我們知道,HashSet 是一個非線程安全的集合.

當在多線程環境下,執行 HashSet的 add方法,如果算哈希槽的時候,如果發生沖突就會導致兩次 add都失敗,從而發生添加元素丟失。

1.5 修復方案

當然解決的方案有很多種,這里采用最簡單的一種解決方案。

將數據初始化部分放到靜態代碼塊中

import java.util.HashSet;
import java.util.Set;/*** @author xing yun*/
public class ConstantUtils {private static final Set<String> goodApiList=new HashSet<>();static {if(goodApiList.isEmpty()){goodApiList.add("A");goodApiList.add("B");goodApiList.add("C");goodApiList.add("D");goodApiList.add("E");goodApiList.add("F");goodApiList.add("G");goodApiList.add("H");goodApiList.add("I");goodApiList.add("J");goodApiList.add("K");}}public static Set<String> getGoodApiList() {return goodApiList;}/*** 靜態工具類應該禁用其構造方法*/private ConstantUtils(){}
}

2 演示示例源碼下載

在這里插入圖片描述

  • 命令行下載
git clone https://github.com/geekxingyun/concurrent-question-fixed-sample.git
  • 訪問 github首頁

https://github.com/geekxingyun/concurrent-question-fixed-sample

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/23529.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/23529.shtml
英文地址,請注明出處:http://en.pswp.cn/web/23529.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

JVM相關:Java內存區域

Java 虛擬機&#xff08;JVM)在執行 Java 程序的過程中會把它管理的內存劃分成若干個不同的數據區域。 Java運行時數據區域是指Java虛擬機&#xff08;JVM&#xff09;在執行Java程序時&#xff0c;為了管理內存而劃分的幾個不同作用域。這些區域各自承擔特定的任務&#xff0c…

Day23 自定義對話框服務

?本章節實現了,自定義對話框服務的功能 當現有的對話框服務無法滿足特定需求時,我們可以采用自定義對話框的解決方案,以更好地滿足一些特殊需求。 一.自定義對話框主機服務步驟 在Models 文件夾中,再建立一個 IDialogHostService 接口類,繼承自 IDialogService 對話框服…

計算兩個LocalDateTime的相差時長

在Java中&#xff0c;你可以使用java.time.Duration類來計算兩個LocalDateTime對象之間的時間差。以下是一個示例代碼&#xff0c;展示了如何計算兩個LocalDateTime實例之間相差的時長&#xff1a; import java.time.Duration; import java.time.LocalDateTime;public class D…

絕對實用Linux命令行下的文件夾逐層創建術,從小白到大神的必學技能

哈嘍&#xff0c;大家好&#xff0c;我是木頭左&#xff01; 基礎篇&#xff1a;初識Linux文件系統 在深入了解如何在Linux中逐層創建文件夾之前&#xff0c;需要對Linux的文件系統有一個基本的認識。Linux文件系統以其樹狀結構而著稱&#xff0c;其中/&#xff08;根目錄&…

實用的供應商管理系統推薦:提升效率的合適選擇

隨著全球化和供應鏈的復雜性增加&#xff0c;供應商管理系統已經成為企業提高運營效率和競爭力的重要工具。一個優秀的供應商管理系統不僅能幫助企業優化采購流程&#xff0c;還能有效地管理供應商關系、降低成本、提高產品質量和服務水平。 供應商管理系統,供應商管理系統推薦…

SIMBA方法解讀

目錄 預處理scRNA-seqscATAC-seq 圖構建&#xff08;5種場景&#xff09;scRNA-seq分析scATAC-seq分析多模態分析批次整合多模態整合 圖學習SIMBA空間中查詢實體識別TF-target genes 預處理 scRNA-seq 過濾掉在少于三個細胞中表達的基因。原始計數按文庫大小標準化&#xff0…

DDS自動化測試落地方案 | 懌星科技攜最新技術亮相是德科技年度盛會

5月28日&#xff0c;懌星科技作為是德科技的重要合作伙伴亮相Keysight World Tech Day 2024。在此次科技盛會上&#xff0c;懌星科技不僅展示了領先的DDS自動化測試解決方案等前沿技術&#xff0c;還分享了在“周期短、任務重”的情況下&#xff0c;如何做好軟件開發和測試驗證…

前端開發之性能優化

本文章 對各大學習技術論壇知識點&#xff0c;進行總結、歸納自用學習&#xff0c;共勉&#x1f64f; 文章目錄 1. [CDN](https://www.bootcdn.cn/)2.懶加載3.緩存4.圖片壓縮5.圖片分割6.sprite7.Code Splitting8.gzip9.GPU加速10.Ajax11.Tree Shaking12.Resource Hints 1. CD…

YOLO系列模型 pt文件轉化為ONNX導出

文章目錄 啥是onnx怎么導出導出之后 啥是onnx Microsoft 和合作伙伴社區創建了 ONNX 作為表示機器學習模型的開放標準。許多框架&#xff08;包括 TensorFlow、PyTorch、scikit-learn、Keras、Chainer、MXNet 和 MATLAB&#xff09;的模型都可以導出或轉換為標準 ONNX 格式。 在…

C++筆試強訓day40

目錄 1.游游的字母串 2.體育課測驗(二) 3.合唱隊形 1.游游的字母串 鏈接https://ac.nowcoder.com/acm/problem/255195 英文字母一共就26個&#xff0c;因此可以直接暴力枚舉以每個字母作為最后的轉變字母。最后去最小值即可 #include <iostream> #include <cmath&…

趕緊收藏!2024 年最常見 20道 Kafka面試題(十)

上一篇地址&#xff1a;趕緊收藏&#xff01;2024 年最常見 20道 Kafka面試題&#xff08;九&#xff09;-CSDN博客 十九、在分布式情況下&#xff0c;Kafka 如何保證消息的順序消費&#xff1f; 在分布式系統中&#xff0c;Kafka保證消息順序消費主要依賴于其分區機制和消費…

項目實戰系列——WebSocket——websock簡介

最近項目中需要用到mes和本地客戶端進行實時通訊&#xff0c;本來想用webapi進行交互的&#xff0c;但是考慮到高效和實時性&#xff0c;就采用這一項技術。 以往采用的方式——長輪詢 客戶端主動向服務器發送一個請求&#xff0c;如果服務器沒有更新的數據&#xff0c;客戶端…

Jtti:docker部署數據庫有哪些優缺點?

在Docker中部署數據庫有其獨特的優缺點。以下是一些主要的優點和缺點&#xff1a; 優點 環境一致性&#xff1a;Docker容器提供了一致的運行環境&#xff0c;從開發到生產環境&#xff0c;確保數據庫運行環境的一致性&#xff0c;減少因環境差異導致的問題。 快速部署和遷移&am…

內置類型知多少?

內置類型&#xff08;也稱為基本類型或原生類型&#xff09;是C/C本身定義的數據類型&#xff0c;它們直接由編譯器支持&#xff0c;不需要用戶自定義。 內置類型主要包括以下幾類&#xff1a; 1&#xff0e;算術類型&#xff1a; (1)整型&#xff1a;int、short、long、lon…

【ARM Cache 系列文章 1.1 -- Cache size 讀取詳細介紹及代碼實現】

請閱讀【ARM Cache 及 MMU/MPU 系列文章專欄導讀】 及【嵌入式開發學習必備專欄】 文章目錄 ARMv8/v9 CPU Cache SizeCache Size 的計算方法Cache Size 讀取代碼實現ARMv8/v9 CPU Cache Size ARM架構通過一系列的系統寄存器來提供CPU和系統的詳細信息,包括緩存的大小和配置。…

五.應用層協議——HTTP協議

HTTP協議 在上一節中&#xff0c;我們提到了協議的本質&#xff0c;其實是雙方約定好的某種格式的數據&#xff0c;常見的就是用結構體或者類來進行表達 而上層的業務邏輯決定了我們協議的定制&#xff0c;有了協議&#xff0c;雙方就可以按照同樣的角度&#xff0c;去解讀數據…

【硬件工程師面試寶典】常見面試題其二

17. 單片機上電后沒有運轉&#xff0c;首先要檢查什么 當單片機上電后沒有運轉時&#xff0c;首先要檢查以下幾方面&#xff1a; 電源電壓&#xff1a;確保電源電壓穩定且符合單片機要求。時鐘信號&#xff1a;檢查時鐘電路是否正常工作&#xff0c;晶振是否振蕩。復位電路&a…

集合體學習01

集合體系結構 Collection 單列集合 Map 雙列集合 Collection 1.List 1.ArrayList 2.LinkedList 3.Vector 2.Set 1.HashSet 1.LinkedHashSet 2.TreeSet 其中Collection&#xff0c;List&#xff0c;Set 為接口&#xff0c;其余為實現類。 List系列集合&#xff1a;添加的元素…

一篇文章帶你入門XXE

1.什么是XXE&#xff1f; XML External Entity&#xff08;XXE&#xff09;攻擊是一種利用 XML 處理器的漏洞&#xff0c;通過引入惡意的外部實體來攻擊應用程序的安全性。這種攻擊通常發生在對用戶提供的 XML 數據進行解析時&#xff0c;攻擊者利用了 XML 規范允許引用外部實體…

kafka-集群搭建(在docker中搭建)

文章目錄 1、kafka集群搭建1.1、下載鏡像文件1.2、創建zookeeper容器并運行1.3、創建3個kafka容器并運行1.3.1、9095端口1.3.2、9096端口1.3.3、9097端口 1.4、重啟kafka-eagle1.5、查看 efak1.5.1、查看 brokers1.5.2、查看 zookeeper 1、kafka集群搭建 1.1、下載鏡像文件 d…