我有以下代码:
public class Experimenter implements Runnable
{
private volatile Integer a = new Integer(0);
public Experimenter() throws Exception
{
System.out.println("start");
}
public void funk() throws InterruptedException
{
synchronized (a)
{
System.out.println("in");
Thread.sleep(5000);
System.out.println("out");
}
}
public static void main(String[] args) throws Exception
{
Thread a = new Thread(new Experimenter(), "a");
Thread b = new Thread(new Experimenter(), "b");
Thread c = new Thread(new Experimenter(), "c");
Thread d = new Thread(new Experimenter(), "d");
Thread e = new Thread(new Experimenter(), "e");
a.start();
b.start();
c.start();
d.start();
e.start();
}
@Override
public void run()
{
try
{
funk();
}
catch (InterruptedException e)
{
e.printStackTrace();
}
}
}
我曾预料到,因为一次只有一个线程可以使用同步块,它会打印如下所示的内容,每个内部之间有5秒的间隙:
start
start
start
start
start
in
out
in
out
in
out
in
out
in
out
但相反,我得到以下内容。所有的ins,5秒后,所有的出局。:
start
start
start
start
start
in
in
in
in
in
out
out
out
out
out
有人可以帮忙解释一下吗?
答案 0 :(得分:3)
非常简单:您的a
之间不会共享Experimenter
;它是一个实例变量,每Experimenter
一个。顺便提一句,volatile
在这种情况下几乎没用。
如果您想要共享锁,请将其设为private static final
变量。在这种情况下也不需要volatile
!
但我会选择@ JBNizet的更清洁的解决方案。
编辑:为什么final
?因为保证在初始化后永远不会改变;但final
变量最重要的一个方面来自Java内存模型,它指出final
变量的初始化发生在任何读取此变量之前。这是一个非常强大的规则。
答案 1 :(得分:2)
每个线程在自己的锁上同步,因为每个线程都使用自己的Experimenter实例,每个Experimenter实例都有自己的Integer实例作为锁。如果希望线程同步,则需要在所有实例之间共享唯一锁。你不应该使用整数,BTW。使用简单的对象:
final Object sharedLock = new Object();
Thread a = new Thread(new Experimenter(sharedLock), "a");
Thread b = new Thread(new Experimenter(sharedLock), "b");
...
使用final关键字: 它不是严格需要的。但锁定,特别是当它们被声明为字段时,应该是最终的,以确保您不会意外地为变量分配新值,这将导致线程使用两个不同的锁定对象
答案 2 :(得分:1)
这肯定会对你有用
public void funk() throws InterruptedException
{
synchronized (Experimenter.class)
{
System.out.println("in");
Thread.sleep(5000);
System.out.println("out");
}
}
答案 3 :(得分:1)
将字段a
声明为static
,它将按预期工作。在您的代码中,该字段是实例成员,因此您实际上有五个不同的监视器。