开门见山看例子,下面是本人学习过程中在网上看到的一段代码及解释
class TestSynchronized extends Thread {
public TestSynchronized(String name) {
super(name);
}
public synchronized static void prt() {
for (int i = 10; i < 20; i++) {
System.out.println(Thread.currentThread().getName() + " : " + i);
try {
Thread.sleep(100);
} catch (InterruptedException e) {
System.out.println("Interrupted");
}
}
}
public synchronized void run() {
for (int i = 0; i < 10; i++) {
System.out.println(Thread.currentThread().getName() + " : " + i);
try {
Thread.sleep(100);
} catch (InterruptedException e) {
System.out.println("Interrupted");
}
}
}
}
public class TestThread {
public static void main(String[] args) {
TestSynchronized t1 = new TestSynchronized("t1");
TestSynchronized t2 = new TestSynchronized("t2");
t1.start();
t1.prt();// (1)
t2.prt();// (2)
}
}
在代码(1)中,虽然是通过对象t1来调用prt()函数的,但由于prt()是静态的,所以调用它时不用经过任何对象,它所属的线程为main线程。
由于调用run()函数取走的是对象锁,而调用prt()函数取走的是class锁,所以同一个线程t1(由上面可知实际上是不同线程)调用run()函数且还没完成run()函数时,它就能调用prt()函数。但prt()函数只能被一个线程调用,如代码(1)和代码(2),即使是两个不同的对象也不能同时调用prt()。