转载

Java并发编程(15):并发编程中实现内存可见的两种方法比较—加锁和volatile变量

在 http://www.importnew.com/?p=20725 这篇博文中,讲述了通过同步实现内存可见性的方法,

在 http://www.importnew.com/?p=20566 这篇博文中,讲述了通过volatile变量实现内存可见性的方法,这里比较下二者的区别。

1、volatile变量是一种稍弱的同步机制在访问volatile变量时不会执行加锁操作,因此也就不会使执行线程阻塞,因此volatile变量是一种比synchronized关键字更轻量级的同步机制。

2、从内存可见性的角度看,写入volatile变量相当于退出同步代码块,而读取volatile变量相当于进入同步代码块。

3、在代码中如果过度依赖volatile变量来控制状态的可见性,通常会比使用锁的代码更脆弱,也更难以理解。仅当volatile变量能简化代码的实现以及对同步策略的验证时,才应该使用它。一般来说,用同步机制会更安全些。

4、加锁机制(即同步机制)既可以确保可见性又可以确保原子性,而volatile变量只能确保可见性,原因是声明为volatile的简单变量如果当前值与该变量以前的值相关,那么volatile关键字不起作用,也就是说如下的表达式都不是原子操作:“count++”、“count = count+1”。

当且仅当满足以下所有条件时,才应该使用volatile变量:

1、对变量的写入操作不依赖变量的当前值,或者你能确保只有单个线程更新变量的值。

2、该变量没有包含在具有其他变量的不变式中。

总结:在需要同步的时候,第一选择应该是synchronized关键字,这是最安全的方式,尝试其他任何方式都是有风险的。尤其在、jdK1.5之后,对synchronized同步机制做了很多优化,如:自适应的自旋锁、锁粗化、锁消除、轻量级锁等,使得它的性能明显有了很大的提升。

本系列:

  • Java并发编程(1):可重入内置锁
  • Java并发编程(2):线程中断(含代码)
  • Java并发编程(3):线程挂起、恢复与终止的正确方法(含代码)
  • Java并发编程(4):守护线程与线程阻塞的四种情况
  • Java并发编程(5):volatile变量修饰符—意料之外的问题(含代码)
  • Java并发编程(6):Runnable和Thread实现多线程的区别(含代码)
  • Java并发编程(7):使用synchronized获取互斥锁的几点说明
  • Java并发编程(8):多线程环境中安全使用集合API(含代码)
  • Java并发编程(9):死锁(含代码)
  • Java并发编程(10):使用wait/notify/notifyAll实现线程间通信的几点重要说明
  • Java并发编程(11):线程间通信中notify通知的遗漏(含代码)
  • Java并发编程(12):线程间通信中notifyAll造成的早期通知问题(含代码)
  • Java并发编程(13):生产者—消费者模型(含代码)
  • Java并发编程(14):图文讲述同步的另一个重要功能—内存可见性
  • Java并发编程(15):并发编程中实现内存可见的两种方法比较—加锁和volatile变量
原文  http://www.importnew.com/20741.html
正文到此结束
Loading...