我正在尝试创建一个小基准(在 Groovy 中),以显示几个同步方法上的高线程争用。当监控自愿上下文切换时,应该会出现高争用,在 Linux 中,这可以通过“pidstat”来实现。
程序如下:
class Res {
private int n;
synchronized public void inc() {
n++;
def foo = []
for (int i = 0; i < 1000; ++i) foo << "hello"
}
synchronized public int getN() {
return n;
}
}
while (true) {
Res res = new Res()
int N = 100000
for (int i = 0; i < N; ++i) {
new Thread({
res.inc()
if (res.getN() == N) {
println "ok"
}
}).start()
}
while (res.getN() < N) {
}
println "========================="
}
但命令
pidstat -w -I -p 26848 5
在自愿上下文切换列上打印 0。该程序创建了 100000 个并发访问同步方法的线程。我不敢相信在这样的工作负载下,没有发生上下文切换。
我的基准有什么问题?
您的命令仅显示主线程的统计信息,不计算子 PID。
Hotspot JVM 有内部同步计数器,但需要一些魔法才能解锁它们:
- Run
jconsole.exe -J-Djconsole.showUnsupported
并连接到您的 JVM。
- Select 连接 -> 热点 MBean -> 创建从主菜单。
- Open sun.management.HotspotRuntime on the MBeans tab.
- You'll find a bunch of counters under InternalRuntimeCounters attribute:
- sun.rt._sync_ContendedLockAttempts
- sun.rt._sync_Parks
- sun.rt._sync_Notifications
- sun.rt._sync_Inflations
etc.
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)