synchronized实现锁的基础:Java中每一个对象都可以作为锁,具体表现为3种形式。
(1)普通同步方法,锁是当前实例对象
(2)静态同步方法,锁是当前类的Class对象
(3)同步方法块,锁是Synchronized括号里配置的对象
首先看一下普通同步方法。
class Sync{
public synchronized void test(String threadname){
System.out.println(threadname+"开始");
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(threadname+"结束");
}
}
class MyThread extends Thread{
public int i;
public MyThread(int i){
this.i=i;
}
public void run(){
Sync sync=new Sync();
sync.test("Thread"+i);
}
}
public class SynTest {
public static void main(String args[]){
for(int i=0;i<3;i++){
Thread thread=new MyThread(i);
thread.start();
}
}
}
运行结果。
Thread2开始 Thread0开始 Thread1开始 Thread2结束 Thread0结束 Thread1结束
虽然test()加了synchronized关键字,但是并没与起作用。这是因为,实例化了3个thread,synchronized加锁的是这3个实例化对象,并不是test()中的代码。需要注意的是,其他线程必须等到该线程释放实例对象之后才能使用该实例化对象。
然后看一下静态同步方法。修改上面的代码。将test()修改为静态的
public static synchronized void test(String threadname)
运行结果
Thread1开始 Thread1结束 Thread0开始 Thread0结束 Thread2开始 Thread2结束
这次synchronized关键字起作用了。这是因为静态同步方法的锁是当前类的Class对象,即Sync类的Class对象。当Thread1获取到该对象执行test()方法,Thread0只有等待Class对象释放之后才能使用。
最后是同步方法块,锁是取决于括号里的配置对象的。
public void test(String threadname){
synchronized(this){
System.out.println(threadname+"开始");
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(threadname+"结束");
}
}
当使用的锁是this,因为我们是为每个线程新建了Sync类,所以它们是3个不同的this。结果如下
Thread0开始 Thread2开始 Thread1开始 Thread0结束 Thread1结束 Thread2结束
当使用Sync.Class对象作为锁时,synchronized的作用就看出来了。同一时刻只能有一个线程执行该段代码。
Thread0开始 Thread0结束 Thread1开始 Thread1结束 Thread2开始 Thread2结束
使用当前类的Class对象作为锁,就相当于对某段代码加了锁。
正在看并发~随手记下来~~参考的This http://blog.csdn.net/xiao__gui/article/details/8188833