基于HashMap多线程并发问题分析

 

问题描述

最近生产系统,某一台业务服务器偶发出现CPU执行90%以上。更严重者导致系统hang掉,事先未找到代码原因,拉取java core里堆栈信息后,发现

发现有几个线程的状态都是Runnable,程序定位在InventoryReportServiceImpl 服务类上,其他的线程都处于Waiting状态

代码定位:

找到该代码位置之后,发现此处代码的业务逻辑为:根据商品SKU去查询对应的商品主数据信息,由于要查询的商品sku数量比较多,所以开发者采取的思想是:通过多线程来进行这批商品的信息查询

实现方式:

通过创建线程池,然后for循环里每一个sku启动线程去查询对应的商品信息,等所有的线程都执行完毕之后,执行后续业务逻辑;这里采用的多线程的计数器来进行控制

初步分析:

每次通过页面点击查询,然后创建线程池,线程池里最大可执行8个子线程,如果sku待查询的商品数量过大的话,那么此过程执行完成耗时较长,此时如果该服务器业务量大的话,那么该报表在执行过程中,由于待处理的sku查询信息数较多,线程池一直处于runnalbe状态;其次:如果用户发现该报表一直未得到相应,就会进行多次点击查询,最后程序会根据多次请求,多次新建线程池,而线程池的多次创建,势必会增加系统消耗,导致CPU过高

进一步分析:

在java core里我们疏忽了一个点:

这句话HashMap.put,根据这条信息,在网上搜索后,发现别人遇到的现象和我们类似

https://www.cnblogs.com/andy-zhou/p/5402984.html

后来根据资料上的例子,进行了本地机器的模拟,通过启动10个异步线程,其中有5个线程都在HashMap里进行put 10000次操作,其余5个进行get操作,然后启动这10个线程,偶现CPU执行到100%

进行本地分析

我们查到消耗CPU最高的进程PID 6024

然后去控制台导出该pid下所有的tid(子线程)

Jstack -l 6024 > c:/6024.txt

 

对pid 进行16进制转换

  1 2018-07-19 11:48:41
  2 Full thread dump Java HotSpot(TM) 64-Bit Server VM (24.65-b04 mixed mode):
  3 
  4 "DestroyJavaVM" prio=6 tid=0x000000001d2fb800 nid=0x3428 waiting on condition [0x0000000000000000]
  5    java.lang.Thread.State: RUNNABLE
  6 
  7    Locked ownable synchronizers:
  8     - None
  9 
 10 "Thread-421" prio=6 tid=0x000000001d177000 nid=0x3828 runnable [0x000000002f4ce000]
 11    java.lang.Thread.State: RUNNABLE
 12     at java.util.HashMap.transfer(Unknown Source)
 13     at java.util.HashMap.resize(Unknown Source)
 14     at java.util.HashMap.addEntry(Unknown Source)
 15     at java.util.HashMap.put(Unknown Source)
 16     at TestLock$2.run(TestLock.java:23)
 17 
 18    Locked ownable synchronizers:
 19     - None
 20 
 21 "Thread-246" prio=6 tid=0x000000001cfe5000 nid=0xcd4 runnable [0x0000000021f0e000]
 22    java.lang.Thread.State: RUNNABLE
 23     at java.util.HashMap.transfer(Unknown Source)
 24     at java.util.HashMap.resize(Unknown Source)
 25     at java.util.HashMap.addEntry(Unknown Source)
 26     at java.util.HashMap.put(Unknown Source)
 27     at TestLock$7.run(TestLock.java:73)
 28 
 29    Locked ownable synchronizers:
 30     - None
 31 
 32 "Thread-244" prio=6 tid=0x000000001cfe3800 nid=0x4e54 runnable [0x0000000021baf000]
 33    java.lang.Thread.State: RUNNABLE
 34     at java.util.HashMap.transfer(Unknown Source)
 35     at java.util.HashMap.resize(Unknown Source)
 36     at java.util.HashMap.addEntry(Unknown Source)
 37     at java.util.HashMap.put(Unknown Source)
 38     at TestLock$5.run(TestLock.java:53)
 39 
 40    Locked ownable synchronizers:
 41     - None
 42 
 43 "Service Thread" daemon prio=6 tid=0x000000000c940800 nid=0x4c98 runnable [0x0000000000000000]
 44    java.lang.Thread.State: RUNNABLE
 45 
 46    Locked ownable synchronizers:
 47     - None
 48 
 49 "C2 CompilerThread1" daemon prio=10 tid=0x000000000af44800 nid=0x3fa8 waiting on condition [0x0000000000000000]
 50    java.lang.Thread.State: RUNNABLE
 51 
 52    Locked ownable synchronizers:
 53     - None
 54 
 55 "C2 CompilerThread0" daemon prio=10 tid=0x000000000af42000 nid=0xfec waiting on condition [0x0000000000000000]
 56    java.lang.Thread.State: RUNNABLE
 57 
 58    Locked ownable synchronizers:
 59     - None
 60 
 61 "Attach Listener" daemon prio=10 tid=0x000000000af40800 nid=0x2aa4 waiting on condition [0x0000000000000000]
 62    java.lang.Thread.State: RUNNABLE
 63 
 64    Locked ownable synchronizers:
 65     - None
 66 
 67 "Signal Dispatcher" daemon prio=10 tid=0x000000000af3f800 nid=0x4118 runnable [0x0000000000000000]
 68    java.lang.Thread.State: RUNNABLE
 69 
 70    Locked ownable synchronizers:
 71     - None
 72 
 73 "Finalizer" daemon prio=8 tid=0x000000000aed4800 nid=0x1ea0 in Object.wait() [0x000000000c3ae000]
 74    java.lang.Thread.State: WAITING (on object monitor)
 75     at java.lang.Object.wait(Native Method)
 76     - waiting on <0x0000000783afd560> (a java.lang.ref.ReferenceQueue$Lock)
 77     at java.lang.ref.ReferenceQueue.remove(Unknown Source)
 78     - locked <0x0000000783afd560> (a java.lang.ref.ReferenceQueue$Lock)
 79     at java.lang.ref.ReferenceQueue.remove(Unknown Source)
 80     at java.lang.ref.Finalizer$FinalizerThread.run(Unknown Source)
 81 
 82    Locked ownable synchronizers:
 83     - None
 84 
 85 "Reference Handler" daemon prio=10 tid=0x000000000aecd000 nid=0x2824 in Object.wait() [0x000000000c21f000]
 86    java.lang.Thread.State: WAITING (on object monitor)
 87     at java.lang.Object.wait(Native Method)
 88     - waiting on <0x0000000783afd210> (a java.lang.ref.Reference$Lock)
 89     at java.lang.Object.wait(Object.java:503)
 90     at java.lang.ref.Reference$ReferenceHandler.run(Unknown Source)
 91     - locked <0x0000000783afd210> (a java.lang.ref.Reference$Lock)
 92 
 93    Locked ownable synchronizers:
 94     - None
 95 
 96 "VM Thread" prio=10 tid=0x000000000aec7800 nid=0x4a14 runnable 
 97 
 98 "GC task thread#0 (ParallelGC)" prio=6 tid=0x000000000280d000 nid=0x2684 runnable 
 99 
100 "GC task thread#1 (ParallelGC)" prio=6 tid=0x000000000280e800 nid=0x4370 runnable 
101 
102 "GC task thread#2 (ParallelGC)" prio=6 tid=0x0000000002810000 nid=0x2d44 runnable 
103 
104 "GC task thread#3 (ParallelGC)" prio=6 tid=0x0000000002812000 nid=0x4898 runnable 
105 
106 "VM Periodic Task Thread" prio=10 tid=0x000000000af4f800 nid=0x4b88 waiting on condition 
107 
108 JNI global references: 144

 

转载于:https://www.cnblogs.com/bailang-LBG/articles/9336751.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值