java 中同步方法和同步代码块的区别详解
java 中同步方法和同步代码块的区别详解
在java语言中,每一个对象有一把锁。线程可以使用synchronized关键字来获取对象上的锁。synchronized关键字可应用在方法级别(粗粒度锁)或者是代码块级别(细粒度锁)。
问题的由来:
看到这样一个面试题:
//下列两个方法有什么区别 public synchronized void method1(){} public void method2(){ synchronized (obj){} }
synchronized用于解决同步问题,当有多条线程同时访问共享数据时,如果进行同步,就会发生错误,java提供的解决方案是:只要将操作共享数据的语句在某一时段让一个线程执行完,在执行过程中,其他线程不能进来执行可以。解决这个问题。这里在用synchronized时会有两种方式,一种是上面的同步方法,即用synchronized来修饰方法,另一种是提供的同步代码块。
这里总感觉怪怪的,这两种方法有什么区别呢,基础学得不好,于是就动手做了个简单的测试,代码如下:
public class synobj { public synchronized void methoda() { system.out.println("methoda....."); try { thread.sleep(5000); } catch (interruptedexception e) { e.printstacktrace(); } } public void methodb() { synchronized(this) { system.out.pritntln("methodb....."); } } public void methodc() { string str = "sss"; synchronized (str) { system.out.println( "methodc....."); } } }
public class testsyn { public static void main(string[] args) { final synobj obj = new synobj(); thread t1 = new thread(new runnable() { @override public void run() { obj.methoda(); } }); t1.start(); thread t2 = new thread(new runnable() { @override public void run() { obj.methodb(); } }); t2.start(); thread t3 = new thread(new runnable() { @override public void run() { obj.methodc(); } }); t3.start(); } }
这段小代码片段打印结果如下:
methoda..... methodc..... //methodb会隔一段时间才会打印出来 methodb.....
这段代码的打印结果是,methoda…..methodc…..会很快打印出来,methodb…..会隔一段时间才打印出来,那么methodb为什么不能像methodc那样很快被调用呢?
在启动线程1调用方法a后,接着会让线程1休眠5秒钟,这时会调用方法c,注意到方法c这里用synchronized进行加锁,这里锁的对象是str这个字符串对象。但是方法b则不同,是用当前对象this进行加锁,注意到方法a直接在方法上加synchronized,这个加锁的对象是什么呢?显然,这两个方法用的是一把锁。
*由这样的结果,我们就知道这样同步方法是用什么加锁的了,由于线程1在休眠,这时锁还没释放,导致线程2只有在5秒之后才能调用方法b,由此,可知两种加锁机制用的是同一个锁对象,即当前对象。
另外,同步方法直接在方法上加synchronized实现加锁,同步代码块则在方法内部加锁,很明显,同步方法锁的范围比较大,而同步代码块范围要小点,一般同步的范围越大,性能就越差,一般需要加锁进行同步的时候,肯定是范围越小越好,这样性能更好*。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!