如何解决寻找令人惊讶的并发Java程序
| 由于我正在编写专注于并发方面的探查器,因此我正在寻找一个使用Java同步机制的良好人工示例。我的探查器使与线程相关的一些动作可见。例如: 呼叫通知/等待 线程更改其状态 一个线程与另一个线程争用监视器锁定 与另一个线程争用之后,该线程已获取了监视器锁 测量每种方法的执行时间 哪个线程访问了某种方法,以及访问频率 等等 因此,我正在寻找的是乍看之下似乎可以理解的Java程序,但是执行该程序时,您会开始怀疑结果。我希望我的探查器能够检测到后台发生了什么。 为了澄清自己,我举一个例子,Brian Goetz的《 Java Concurrency in Practice》一书给出了“有毒”代码示例,这些示例用于学习。@NotThreadSafe
public class ListHelper<E> {
public List<E> list =
Collections.synchronizedList(new ArrayList<E>());
...
public synchronized boolean putIfAbsent(E x) {
boolean absent = !list.contains(x);
if (absent)
list.add(x);
return absent;
}
}
它旨在通过方法“ 1”作为线程安全类的扩展。由于list
已同步,但putIfAbsent
使用另一个锁来保护状态,如列表中定义的方法。
探查器可以显示已使用的监视器锁,并且让用户感到惊讶(或不...),用户将看到有两个可能的监视器锁而不是一个。
我不太喜欢这个例子,但是我不会问我是否已经有了很多很好的例子。
我发现我的问题与此类似:Java中最常见的并发问题是什么?和Java并发错误模式。
但是它们仅指损坏的并发程序。我也在寻找线程安全的实现,但是仍然不明显它们是线程安全的。
解决方法
查看FindBugs错误描述列表,特别是属于多线程正确性类别的列表(右表列)。
这些错误中的每一个都包含有关特定习语不好的原因以及如何解决的参考。
, 我会回到过去,大概是七年或更久,然后找到java.util.concurrent之前那个时代的一些开放源代码。几乎所有采用自身并发的事物都会有一些细微的错误,因为并发很难实现。
, 这个怎么样?
class ObjectReference {
private volatile Object obj = null;
public void set(Object obj) {
if (obj == null) {
throw new IllegalArgumentException();
}
this.obj = obj;
synchronized (this) {
notifyAll();
}
}
/**
* This method never returns null
*/
public Object waitAndGet() {
if (obj != null) {
return obj;
}
synchronized (this) {
wait();
return obj;
}
}
}
您实际上可以从waitAndGet()
得到null
。请参阅-实际是否发生虚假唤醒?
, 餐饮哲学家的问题就是经典的并发例子。该链接有一个可能的解决方案,您可以在网上找到更多的解决方案。
如第一个链接中所述,此示例说明了许多常见的并发问题。请让您的探查器显示它可以跟踪多少个!
, 请参阅Java专家通讯,以获取源源不断的小型Java难题,其中许多难题都可以满足您的测试需求。
, 我建议四处逛逛(或询问作者)IBM ConTest基准套件,因为它包含许多Java并发性错误(不幸的是,它们不是大型的开源程序)。关于此基准的好处是,已经记录了错误(类型和位置)。
如果您想找到更多程序,我建议您看一下软件测试/并发程序质量方面的一些研究论文。他们应该指出他们在研究中使用的示例程序。
如果所有其他方法都失败了,您可以尝试在GitHub(或类似服务)上搜索包含必要的并发机制(即同步)的存储库。您可能会以这种方式找到大量Java代码,唯一的问题是未记录错误(除非您查找提交修复程序)。
我认为这三个建议将为您提供足够的程序来测试并发分析器。
, 也许是Eclipse还是Tomcat部署?两者都不是很人为,但是我可以想象在调试一个或另一个时想要好的工具。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。