Synchronized同步方法及Synchronized同步语句块

1 Synchronized同步方法

1.1 述

  • **“线程安全”就是以获得的实例变量的值是经过同步处理的,不会出现脏读的现象。
  • “非线程安全”问题存在于“实例变量”中,如果是方法内部的私有变量,则不存在“非线程安全”问题,所得结果也就是“线程安全”的了。
  • 方法中的变量不存在非线程安全问题,永远都是线程安全的。这是方法内部的变量是私有的特性造成的。
  • 如果多个线程共同访问1个对象中的实例变量,则有可能出现“非线程安全”问题。
  • 两个线程同时访问一个没有同步的方法,如果两个线程同时操作业务对象中的实例变量,则有可能会出现“非线程安全”问题
  • 在两个线程访问同一个对象中的同步方法时一定是线程安全的。
  • 两个线程分别访问同一个类的两个不同实例的相同名称的同步方法,效果却是以异步的方式运行的.
    关键字synchronized取得的锁都是对象锁,而不是把一段代码或方法(函数)当作锁,所以在上面的示例中,哪个线程先执行带synchronized关键字的方法,哪个线程就持有该方法所属对象的锁Lock,那么其他线程只能呈等待状态,前提是多个线程访问的是同一个对象。
  • 但如果多个线程访问多个对象,则JVM会创建多个锁。调用用关键字synchronized声明的方法一定是排队运行的。
  • 虽然线程A先持有了object对象的锁,但线程B完全可以异步调用非synchronized类型的方法。
  • 1)A线程先持有object对象的Lock锁,B线程可以以异步的方式调用object对象中的非synchronized类型的方法。
  • 2)A线程先持有object对象的Lock锁,B线程如果在这时调用object对象中的synchronized类型的方法则需等待,也就是同步。
  • 发生脏读的情况是在读取实例变量时,此值已经被其他线程更改过了。

1.2 synchronized拥有锁重入

  • 关键字synchronized拥有锁重入的功能,也就是在使用synchronized时,当一个线程得到一个对象锁后,再次请求此对象锁时是可以再次得到该对象的锁的。这也证明在一个synchronized方法/块的内部调用本类的其他synchronized方法/块时,是永远可以得到锁的
  • 可重入锁”的概念是:自己可以再次获取自己的内部锁。比如有1条线程获得了某个对象的锁,此时这个对象锁还没有释放,当其再次想要获取这个对象的锁的时候还是可以获取的,如果不可锁重入的话,就会造成死锁。 可重入锁也支持在父子类继承的环境中
  • 当存在父子类继承关系时,子类是完全可以通过“可重入锁”调用父类的同步方法的。

1.3 出现异常,锁自动释放

  • 当一个线程执行的代码出现异常时,其所持有的锁会自动释放。

1.4 同步不具有继承性

  • 同步不能继承,所以还得在子类的方法中添加synchronized关键字

2 synchronized同步代码块

  • 当两个并发线程访问同一个对象object中的synchronized(this)同步代码块时,一段时间内只能有一个线程被执行,另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块。
  • 当一个线程访问object的一个synchronized同步代码块时,另一个线程仍然可以访问该object对象中的非synchronized(this)同步代码块。
  • 在使用同步synchronized(this)代码块时需要注意的是,当一个线程访问object的一个synchronized(this)同步代码块时,其他线程对同一个object中所有其他synchronized(this)同步代码块的访问将被阻塞,这说明synchronized使用的“对象监视器”是一个
  • synchronized方法一样,synchronized(this)代码块也是锁定当前对象的。
  • 多个线程调用同一个对象中的不同名称的synchronized同步方法或synchronized(this)同步代码块时,调用的效果就是按顺序执行,也就是同步的,阻塞的。

2.2 对任意对象监视

  • 使用synchronized(this)格式来同步代码块,其实Java还支持对“任意对象”作为“对象监视器”来实现同步的功能。这个“任意对象”大多数是实例变量及方法的参数使用格式为synchronized(非this对象).
    作用只有1种:
  • synchronized(非this对象x)同步代码块。
  • 1)在多个线程持有“对象监视器”为同一个对象的前提下,同一时间只有一个线程可以执行synchronized(非this对象x)同步代码块中的代码。
  • 2)当持有“对象监视器”为同一个对象的前提下,同一时间只有一个线程可以执行synchronized(非this对象x)同步代码块中的代码。
  • 如果在一个类中有很多个synchronized方法,这时虽然能实现同步,但会受到阻塞,所以影响运行效率;> * 但如果使用同步代码块锁非this对象,则synchronized(非this)代码块中的程序与同步方法是异步的,不与其他锁this同步方法争抢this锁,则可大大提高运行效率。
  • 使用“synchronized(非this对象x)同步代码块”格式进行同步操作时,对象监视器必须是同一个对象。如果不是同一个对象监视器,运行的结果就是异步调用了,就会交由于对象监视器不同,所以运行结果就是异步的
  • synchronized(非this对象x)”格式的写法是将x对象本身作为“对象监视器”。
    这样就可以得出以下3个结论:
  • 1)当多个线程同时执行synchronized(x){}同步代码块时呈同步效果。
  • 2)当其他线程执行x对象中synchronized同步方法时呈同步效果。
  • 3)当其他线程执行x对象方法里面的synchronized(this)代码块时也呈现同步效果。 但需要注意:如果其他线程调用不加synchronized关键字的方法时,还是异步调用。

静态同步synchronized方法与synchronized(class)代码块

  • 关键字synchronized还可以应用在static静态方法上,如果这样写,那是对当前的*.java文件对应的Class类进行持锁。
  • synchronized关键字加到static静态方法上是给Class类上锁,而synchronized关键字加到非static静态方法上是给对象上锁。
  • 异步的原因是持有不同的锁,一个是对象锁,另外一个是Class锁,而Class锁可以对类的所有对象实例起作用
  • 同步synchronized(class)代码块的作用其实和synchronized static方法的作用一样。

将synchronized(string)同步块与String联合使用时,要注意常量池以带来的一些例外。

你可能感兴趣的:(Synchronized同步方法及Synchronized同步语句块)