[Java多线程] volatile 关键字正确使用方法

volatile 变量具有 synchronized 的可见性特性,但是不具备原子特性,即多线程环境中,使用 volatile 关键字的变量仅可以保证不同线程读取变量时,可以读到最新修改的变量值,但是修改变量值时,却不能保证线程安全(可能存在写值覆盖现象)。以下测试代码,展示了使用volatile关键字的几种方式。

 1 /**
 2  * <b>volatile 关键字正确用法</b><br>
 3  * @author        Gaylen
 4  * @version        V1.1.0
 5  * history
 6  * 1.1.0, 2014年11月20日        Gaylen            FE
 7  * @since        Java 6.0
 8  */
 9 public class TestVolatile {
10
11     /** volatile + atomic 保证读写安全 */
12     public static volatile AtomicInteger count1 = new AtomicInteger(0);
13
14     /** volatile 实现读安全,但不能保证写安全 */
15     public static volatile int count2 = 0;
16
17     /** volatile + synchronized 实现读写安全 */
18     public static volatile int count3 = 0;
19
20     /** static 仅保证全局唯一,但不能保证读写安全 */
21     public static int count4 = 0;
22
23     public static synchronized void count3Increment() {
24         TestVolatile.count3++;
25     }
26
27     /** 测试线程数 */
28     public static final int numOfThread = 1000;
29
30     /** 线程辅助类,保证所有线程执行完毕 */
31     private static CountDownLatch countDownLatch = new CountDownLatch(numOfThread);
32
33     public static void increment() {
34         try {
35             Thread.sleep(1);
36         } catch (InterruptedException e) {
37         }
38         count1.getAndIncrement();
39         count2++;
40         count3Increment();
41         count4++;
42     }
43
44     /**
45      * 输出结果
46      * 等待所有线程执行完毕后,输出结果
47      */
48     public static void print() {
49         try {
50             countDownLatch.await();
51         } catch (InterruptedException e) {
52             e.printStackTrace();
53         }
54         System.out.println("运行结果: count1=" + TestVolatile.count1.get());
55         System.out.println("运行结果: count2=" + TestVolatile.count2);
56         System.out.println("运行结果: count3=" + TestVolatile.count3);
57         System.out.println("运行结果: count4=" + TestVolatile.count4);
58         System.out.println("---------------------------------------");
59
60     }
61
62     /**
63      * <b>程序入口</b><br>
64      * 同时启动1000个线程,进行增加操作
65      * @param args
66      */
67     public static void main(String[] args) {
68         for (int i = 0; i < numOfThread; i++) {
69             new Thread(new Runnable() {
70
71                 @Override
72                 public void run() {
73                     for (int index = 0; index < 1000; index++) {
74                         TestVolatile.increment();
75                     }
76                     countDownLatch.countDown();
77                 }
78             }).start();
79         }
80         print();
81     }
82 }

测试程序输出结果如下:

运行结果: count1=1000000
运行结果: count2=998528
运行结果: count3=1000000
运行结果: count4=999892
---------------------------------------

通过测试程序可知,使用 volatile + synchronized 方式 或者 volatile + 原子变量 方式可以同时保证多线程环境下安全的变量读写。

时间: 2024-11-08 04:23:20

[Java多线程] volatile 关键字正确使用方法的相关文章

Java多线程——volatile关键字、发布和逸出

1.volatile关键字 Java语言提供了一种稍弱的同步机制,即volatile变量.被volatile关键字修饰的变量不会被缓存在寄存器或者对其他处理器不可见的地方,因此在每次读取volatitle类型变量时总会返回最新的值,而不是从寄存器中获取. 加锁机制既然可以确保可见性又可以确保原子性,而volatile只能确保可见性. 2.发布和逸出 "发布(Publish)"一个对象指,使对象能够在当前作用域之外的代码中使用.如将指向该对象的引用保存到其他代码可以访问的地方,或者在某一

java多线程 -- volatile 关键字 内存 可见性

内存可见性(Memory Visibility) 1 内存可见性(Memory Visibility)是指当某个线程正在使用对象状态而另一个线程在同时修改该状态,需要确保当一个线程修改了对象   状态后,其他线程能够看到发生的状态变化. 2 可见性错误是指当读操作与写操作在不同的线程中执行时,我们无法确保执行读操作的线程能适时地看到其他线程写入的值,有时甚至是根本不可能的事情.   我们可以通过同步来保证对象被安全地发布.除此之外我们也可以使用一种更加轻量级的 volatile 变量. vola

Java多线程volatile关键字

JVM中,分为JVM Client模式,JVM Server模式.两者区别并不大,但serve启动模式启动较慢,但一旦运行起来后,在hash和method call方面的效率,比Client模式效率高数十倍. 线程在读取数据时,首先从主内存中读取数据,保存在本地工作内存中,然后对数据进行修改,修改完毕后,写入本地工作内存中,再从工作内存写入主内存.示例图如下: JVM如果以Server模式启动时,为了线程的效率,线程在第一次读取数据后,将数据保存在工作内存中,之后读取数据时,均从工作内存中读取.

java中volatile关键字

java中volatile关键字 学习ImageLoader源码时遇到遇到这么一个关键字,以前没遇到过,也不明白是何含义,网上找了部分资料,总结记录一下. 在java线程并发处理中,有一个关键字volatile的使用目前存在很大的混淆,以为使用这个关键字,在进行多线程并发处理的时候就可以万事大吉.Java语言是支持多线程的,为了解决线程并发的问题,在语言内部引入了 同步块 和 volatile 关键字机制. synchronized 同步块大家都比较熟悉,通过 synchronized 关键字来

转:java中volatile关键字的含义

转:java中volatile关键字的含义 在java线程并发处理中,有一个关键字volatile的使用目前存在很大的混淆,以为使用这个关键字,在进行多线程并发处理的时候就可以万事大吉. Java语言是支持多线程的,为了解决线程并发的问题,在语言内部引入了 同步块 和 volatile 关键字机制. synchronized 同步块大家都比较熟悉,通过 synchronized 关键字来实现,所有加上synchronized 和 块语句,在多线程访问的时候,同一时刻只能有一个线程能够用 sync

【转】java中volatile关键字的含义

java中volatile关键字的含义 在java线程并发处理中,有一个关键字volatile的使用目前存在很大的混淆,以为使用这个关键字,在进行多线程并发处理的时候就可以万事大吉. Java语言是支持多线程的,为了解决线程并发的问题,在语言内部引入了 同步块 和 volatile 关键字机制. synchronized 同步块大家都比较熟悉,通过 synchronized 关键字来实现,所有加上synchronized 和 块语句,在多线程访问的时候,同一时刻只能有一个线程能够用 synchr

Java中volatile关键字的作用

在Java内存模型中,有main memory(主内存)还每个线程各自的线程内存memory(例如:寄存器).为了性能一个线程会在自己memory中保持要访问变量的副本.这样就会出现同一个变量在某一个时刻一个线程内存中的值和其他线程内存或者主内存中的值不一致. 一个变量声明为volatile,就意味着这个变量随时会被其他线程修改,因此不能将他cahe在线程memory中,即:不会再memory中保留他的副本,下面看个例子: public class TestVolatile extends Th

Java中volatile关键字解析

一.内存模型的相关概念 大家都知道,计算机在执行程序时,每条指令都是在CPU中执行的,而执行指令过程中,势必涉及到数据的读取和写入.由于程序运行过程中的临时数据是存放在主存(物理内存)当中的,这时就存在一个问题,由于CPU执行速度很快,而从内存读取数据和向内存写入数据的过程跟CPU执行指令的速度比起来要慢的多,因此如果任何时候对数据的操作都要通过和内存的交互来进行,会大大降低指令执行的速度.因此在CPU里面就有了高速缓存. 也就是,当程序在运行过程中,会将运算需要的数据从主存复制一份到CPU的高

java中volatile关键字的理解

一.基本概念 Java 内存模型中的可见性.原子性和有序性.可见性: 可见性是一种复杂的属性,因为可见性中的错误总是会违背我们的直觉.通常,我们无法确保执行读操作的线程能适时地看到其他线程写入的值,有时甚至是根本不可能的事情.为了确保多个线程之间对内存写入操作的可见性,必须使用同步机制. 可见性,是指线程之间的可见性,一个线程修改的状态对另一个线程是可见的.也就是一个线程修改的结果.另一个线程马上就能看到.比如:用volatile修饰的变量,就会具有可见性.volatile修饰的变量不允许线程内