这个练习直接来自 Kathy Seirra 和 Bert Bates 的 SCJP
同步代码块
在本练习中,我们将尝试同步一段代码。在该代码块中,我们将获得一个对象的锁,以便其他线程在代码块执行时无法修改它。我们将创建三个线程,它们都将尝试操作同一个对象。每个线程将输出一个字母 100 次,然后将该字母加一。我们将使用的对象是 StringBuffer。
我们可以在 String 对象上进行同步,但字符串一旦创建就无法修改,因此如果不生成新的 String 对象,我们将无法增加字母。最终输出应该有 100 个 As、100 个 Bs 和 100 个 Cs,所有这些都在不间断的行中。
- 创建一个类并扩展 Thread 类。
- 重写 Thread 的 run() 方法。这就是同步代码块的去向。
- 为了让我们的三个线程对象共享同一个对象,我们需要创建一个构造函数,该构造函数在参数中接受一个 StringBuffer 对象。
- 同步代码块将从第 3 步获得对 StringBuffer 对象的锁定。
- 在块内,输出 StringBuffer 100 次,然后递增 StringBuffer 中的字母。你可以查看第 6 章的 StringBuffer 方法来帮助解决这个问题。
- 最后,在 main() 方法中,使用字母 A 创建一个 StringBuffer 对象,然后创建我们类的三个实例并启动所有三个实例。
我已经为上述练习编写了以下课程(而不是 100 我正在打印 10 个字符)
class MySyncBlockTest extends Thread {
StringBuffer sb;
MySyncBlockTest(StringBuffer sb) {
this.sb=sb;
}
public static void main (String args[]) {
StringBuffer sb = new StringBuffer("A");
MySyncBlockTest t1 = new MySyncBlockTest(sb);
MySyncBlockTest t2 = new MySyncBlockTest(sb);
MySyncBlockTest t3 = new MySyncBlockTest(sb);
t1.start();
t2.start();
t3.start();
}
public void run() {
synchronized(this) {
for (int i=0; i<10; i++) {
System.out.print(sb);
}
System.out.println("");
if (sb.charAt(0)=='A')
sb.setCharAt(0, 'B');
else
sb.setCharAt(0, 'C');
}
}
}
我期待一个类似于以下的输出(10 As,10 Bs 和 10 Cs),但没有得到它。
AAAAAAAAAA
BBBBBBBBBB
CCCCCCCCCC
相反,我得到了如下不同的输出,因为三个线程有机会在另一个线程完成之前进入循环。
AAAAAAAAAAAAAAAAAA
ABB
ACCCCCCCC
我的问题是为什么 run 方法中的同步块不起作用?