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.mzph.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 对话框服…

绝对实用Linux命令行下的文件夹逐层创建术,从小白到大神的必学技能

哈喽&#xff0c;大家好&#xff0c;我是木头左&#xff01; 基础篇&#xff1a;初识Linux文件系统 在深入了解如何在Linux中逐层创建文件夹之前&#xff0c;需要对Linux的文件系统有一个基本的认识。Linux文件系统以其树状结构而著称&#xff0c;其中/&#xff08;根目录&…

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&…

项目实战系列——WebSocket——websock简介

最近项目中需要用到mes和本地客户端进行实时通讯&#xff0c;本来想用webapi进行交互的&#xff0c;但是考虑到高效和实时性&#xff0c;就采用这一项技术。 以往采用的方式——长轮询 客户端主动向服务器发送一个请求&#xff0c;如果服务器没有更新的数据&#xff0c;客户端…

五.应用层协议——HTTP协议

HTTP协议 在上一节中&#xff0c;我们提到了协议的本质&#xff0c;其实是双方约定好的某种格式的数据&#xff0c;常见的就是用结构体或者类来进行表达 而上层的业务逻辑决定了我们协议的定制&#xff0c;有了协议&#xff0c;双方就可以按照同样的角度&#xff0c;去解读数据…

一篇文章带你入门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…

实时监控电脑屏幕软件有哪些?(珍藏篇)

在当今的数字化工作环境中&#xff0c;实时监控电脑屏幕软件是企业管理、远程协助、教育监控等领域不可或缺的工具。 这些软件能够帮助管理者了解员工的工作状态、提升团队协作效率、确保数据安全&#xff0c;同时在家庭教育和远程技术支持中也有广泛应用。 以下是精选的几款实…

创意SQL,高考祝福!一起为学子们加油助威!

&#x1f4e2;&#x1f4e2;&#x1f4e2;&#x1f4e3;&#x1f4e3;&#x1f4e3; 作者&#xff1a;IT邦德 中国DBA联盟(ACDU)成员&#xff0c;10余年DBA工作经验&#xff0c; Oracle、PostgreSQL ACE CSDN博客专家及B站知名UP主&#xff0c;全网粉丝10万 擅长主流Oracle、My…

X-Caps

用于对视觉属性进行编码的胶囊 补充信息 数据集太大&#xff0c;不建议复现

【西瓜书】4.决策树

1 递归返回情况 &#xff08;1&#xff09;结点包含样本全为同一类别 &#xff08;2&#xff09;属性集为空&#xff0c;没有属性可供划分了 或 有属性&#xff0c;但是在属性上划分的结果都一样 &#xff08;3&#xff09;结点为空结点 **结束时判定该结点的类别遵循如下规则&…

『哈哥赠书 - 54期』-『架构思维:从程序员到CTO』

文章目录 ⭐️ 架构思维&#xff1a;从程序员到CTO⭐️ 本书简介⭐️ 作者简介⭐️ 编辑推荐⭐️ 不想成为架构师的程序员不是好CTO 在程序员的职业规划中&#xff0c;成为软件架构师是一个非常有吸引力的选择。但是对于如何才能成为一名架构师&#xff0c;不少同学认为只要代码…

uniadmin引入iconfont报错

当在uniadmin中引入iconfont后&#xff0c;出现错误&#xff1a; [plugin:vite:css] [postcss] Cannot find module ‘E:/UniAdmin/uniAdmin/static/fonts/iconfont.woff2?t1673083050786’ from ‘E:\UniAdmin\uniAdmin\static\fonts\iconfont.css’ 这是需要更改为绝对路径…

王炸级产品:字节跳动的Seed-TTS

在人工智能的快速发展中&#xff0c;文本到语音&#xff08;TTS&#xff09;技术已成为连接数字世界与人类沟通的重要桥梁。而字节跳动推出的Seed-TTS模型&#xff0c;无疑是这一领域的一个突破性进展&#xff0c;它以其卓越的性能和高度的自然度&#xff0c;被誉为TTS模型中的…

如何快速分析并将一个简单的前后端分离项目跑起来

一、前言 主要是前一段时间有小伙伴问我说自己刚入坑学后端不久&#xff0c;在开源网站上找了个简单的前后端分离项目&#xff0c;但是自己不会跑起来&#xff0c;让我给他说说&#xff0c;介于这玩意三两句话不是很好说清楚&#xff0c;而且不清楚那个小伙伴的知识到何种地步…