同步访问共享的可变数据(synchronized与volatile关键字)

synchronized 关键字可以保证同一时刻,只有一个线程可以执行某一个方法,或是某一个代码块。

成都一家集口碑和实力的网站建设服务商,拥有专业的企业建站团队和靠谱的建站技术,十多年企业及个人网站建设经验 ,为成都上千家客户提供网页设计制作,网站开发,企业网站制作建设等服务,包括成都营销型网站建设,成都品牌网站建设,同时也为不同行业的客户提供做网站、成都网站制作的服务,包括成都电商型网站制作建设,装修行业网站制作建设,传统机械行业网站建设,传统农业行业网站制作建设。在成都做网站,选网站制作建设服务商就选成都创新互联公司

它包含两个特征:1、互斥 2、可见。即同步不仅可以阻止一个线程看到对象处于不一致的状态中,还可以保证进入同步方法或者同步代码块的每个线程,都看到由同一个锁保护的之前所有的修改效果。

java语言规范保证读或者写一个变量时原子的,除非这个变量的类型为long或者double。

读取一个非long或double类型的变量,可以保证返回的值是某个线程保存在该变量中的,即使多线程在没有同步的情况下并发的修改这个变量也是如此。

虽然语言规范保证了线程在读取原子数据的时候,不会看到任意的数值,但是它并不保证一个线程写入的值对于另一个线程是可见的。为了在线程之间进行可靠通信,也为了互斥访问,同步是必要的。

Java代码

 
 
 
  1. public class StopThread {     
  2.     private static boolean stopRequested = false;     
  3.     
  4.     public static synchronized boolean isStopRequested() {     
  5.         return stopRequested;     
  6.     }     
  7.     
  8.     public static synchronized void setStopRequested(boolean stopRequested) {     
  9.         StopThread.stopRequested = stopRequested;     
  10.     }     
  11.     
  12.     public static void main(String[] args) {     
  13.         try {     
  14.             new Thread(new Runnable() {     
  15.                 @Override    
  16.                 public void run() {     
  17.                     int i = 0;     
  18.                     while (!isStopRequested()) {     
  19.                         System.out.println(i++);     
  20.                     }     
  21.                 }     
  22.             }).start();     
  23.     
  24.             TimeUnit.SECONDS.sleep(1);     
  25.         } catch (InterruptedException e) {     
  26.             e.printStackTrace();     
  27.         }     
  28.         setStopRequested(true);     
  29.     }     
  30. }  

 

上面的synchronized关键字是需要的,如果没有同步的话,这个程序永远不会终止:因为不能保证后台线程何时"看到"主线程对stopRequested的值所做的改变,后台线程永远在循环。

注意:读写方法都要被同步,否则同步就不会起作用。

stopRequested即使没有被同步也是原子的,这些同步方法是为了它的 通信效果 ,而不是为了互斥访问。

volatile 变量可以被看作是一种 “程度较轻的 synchronized”;与 synchronized 块相比,volatile 变量所需的编码较少,并且运行时开销也较少,但是它所能实现的功能也仅是 synchronized 的一部分。

锁提供了两种主要特性:互斥(mutual exclusion) 和可见性(visibility)。互斥即一次只允许一个线程持有某个特定的锁,因此可使用该特性实现对共享数据的协调访问协议,这样,一次就只有一个线程能够使用该共享数据。可见性要更加复杂一些,它必须确保释放锁之前对共享数据做出的更改对于随后获得该锁的另一个线程是可见的 —— 如果没有同步机制提供的这种可见性保证,线程看到的共享变量可能是修改前的值或不一致的值,这将引发许多严重问题。

Volatile 变量具有 synchronized 的可见性特性,但是不具备原子特性。这就是说线程能够自动发现 volatile 变量的***值。Volatile 变量可用于提供线程安全,但是只能应用于非常有限的一组用例:多个变量之间或者某个变量的当前值与修改后值之间没有约束。

Java代码

 
 
 
  1. public class StopThread2 {     
  2.     private static volatile boolean stopRequested = false;     
  3.     
  4.     public static boolean isStopRequested() {     
  5.         return stopRequested;     
  6.     }     
  7.     
  8.     public static void setStopRequested(boolean stopRequested) {     
  9.         StopThread2.stopRequested = stopRequested;     
  10.     }     
  11.     
  12.     public static void main(String[] args) {     
  13.         try {     
  14.             new Thread(new Runnable() {     
  15.                 @Override    
  16.                 public void run() {     
  17.                     int i = 0;     
  18.                     while (!isStopRequested()) {     
  19.                         System.out.println(i++);     
  20.                     }     
  21.                 }     
  22.             }).start();     
  23.     
  24.             TimeUnit.SECONDS.sleep(1);     
  25.         } catch (InterruptedException e) {     
  26.             e.printStackTrace();     
  27.         }     
  28.         setStopRequested(true);     
  29.     }     
  30. }   

 

单独使用 volatile 还不足以实现计数器,问题在于操作符(++)不是原子的,例如

Java代码

 
 
 
  1. private static volatile int nextSerialNumber = 0;     
  2. public static int generaterSerialNumber(){     
  3.        return nextSerialNumber ++;     
  4. }   

 

它在nextSerialNumber域中执行两个操作:首先它读取值,然后写回一个新值,相当于原来的值再加上1。如果第二个线程在***个线程读取旧值和写回新值期间读取这个域,第二个线程就会与***个线程看到同一值,并返回相同的序列号,这个程序会计算出错误结果。

修正generaterSerialNumber的方法的一种方法是:在它的声明中去掉volatile增加synchronized修饰符。这样可以确保多个调用不会交叉存取,确保每个调用都会看到之前所有调用的效果。

***的修正方法是:使用类AtomicLong

Java代码

 
 
 
  1. private static final AtomicLong nextSerialNumber = new AtomicLong();     
  2. public static long generaterSerialNumber(){     
  3.        return nextSerialNumber.getAndIncrement();     
  4. }  

 

简而言之,多个线程共享可变数据的时候,每个读或写数据的线程都必须执行同步。如果没有同步,就无法保证一个线程所做的修改可以被另一个线程获知。如果需要线程之间的交互通信,而不需要互斥,volatile修饰符就是一种可以接受的形式,但需要正确的使用。

本文题目:同步访问共享的可变数据(synchronized与volatile关键字)
标题网址:http://www.csdahua.cn/qtweb/news4/302054.html

网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网