java中的synchronized(同步代码块和同步方法的区别)

问题的由来:

看到这样一个面试题:java

//下列两个方法有什么区别
public synchronized void method1(){}

public void method2(){
 synchronized  (obj){}
}

复制代码

synchronized用于解决同步问题,当有多条线程同时访问共享数据时,若是不进行同步,就会发生错误,java提供的解决方案是:只要将操做共享数据的语句在某一时段让一个线程执行完,在执行过程当中,其余线程不能进来执行能够。解决这个问题。这里在用synchronized时会有两种方式,一种是上面的同步方法,即用synchronized来修饰方法,另外一种是提供的同步代码块。面试

这里总感受怪怪的,这两种方法有什么区别呢,基础学得很差,因而就动手作了个简单的测试,代码以下:bash

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();
	}
}
复制代码

这段小代码片断打印结果以下:ide

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,由此,可知两种加锁机制用的是同一个锁对象,即当前对象。this

另外,同步方法直接在方法上加synchronized实现加锁,同步代码块则在方法内部加锁,很明显,同步方法锁的范围比较大,而同步代码块范围要小点,通常同步的范围越大,性能就越差,通常须要加锁进行同步的时候,确定是范围越小越好,这样性能更好spa

相关文章
相关标签/搜索