在进行多线程开发时,经常会用到同步锁:synchronized。该关键字可以使用在普通方法上,也可以使用在static方法上。这里叙述下这两种方式的区别,避免以后忘记。
我们都知道,每一个对象都有一个监视器。对于每一个类,有一个Class对象,也可以有该类的实例对象。如果synchronized放在普通方法上,线程在同步访问时,就会查看该类实例对象上的监视器;如果synchronized放在static方法上,则会查看Class对象上的监视器。因为是两个不同的对象,所以这两种方式不会互相干扰,线程可以并发访问。
看下面这个例子:
public class TestSynchronizedA { public synchronized static void m1() { System.out.println("m1"); } public synchronized void m2() { System.out.println("m2"); } public static void main(String[] args) { final TestSynchronizedA A = new TestSynchronizedA(); Executor exec = Executors.newCachedThreadPool(); exec.execute(new Runnable() { @Override public void run() { A.m2(); } }); exec.execute(new Runnable() { @Override public void run() { TestSynchronizedA.m1(); A.m1(); } }); } }
m1是static方法,m2是普通方法。synchronized关键字加在这俩方法上,因此,在线程并发访问时,对于m1,使用的是Class对象的监视器,对于m2,使用的是普通对象的监视器(如例子中的对象A)。
例子中的两个线程在执行时,可以同时并发访问m1和m2。在使用普通对象A调用static方法m1时,也是可以并发访问的。
因此,总结如下:
1、对于静态方法,其使用的是Class上的监视器,无论使用何种调用方式(类名调用静态方法或实例调用静态方法)。
2、对于普通方法,其使用的是实例上的监视器。
3、对于静态方法和普通方法,线程对他们之间的访问没有任何限制。但如果访问同类型的方法(如,同时访问多个静态方法或同时访问多个普通方法),会有同步锁的限制(前提:访问的是同一个对象)。
原文地址:http://blog.csdn.net/ado1986/article/details/41205643