`
langgufu
  • 浏览: 2306922 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

volatile失效,慎重使用volatile关键字

阅读更多

volatile关键字相信了解Java多线程的读者都很清楚它的作用。volatile关键字用于声明简单类型变量,如int、float、boolean等数据类型。如果这些简单数据类型声明为volatile,对它们的操作就会变成原子级别的。但这有一定的限制。例如,下面的例子中的n就不是原子级别的:

  1. package mythread;  
  2.  
  3. public class JoinThread extends Thread  
  4. {  
  5.     public static volatile int n = 0;  
  6.     public void run()  
  7.     {  
  8.         for (int i = 0; i < 10; i++)  
  9.             try 
  10.         {  
  11.                 n = n + 1;  
  12.                 sleep(3); // 为了使运行结果更随机,延迟3毫秒  
  13.  
  14.             }  
  15.             catch (Exception e)  
  16.             {  
  17.             }  
  18.     }  
  19.  
  20.     public static void main(String[] args) throws Exception  
  21.     {  
  22.  
  23.         Thread threads[] = new Thread[100];  
  24.         for (int i = 0; i < threads.length; i++)  
  25.             // 建立100个线程  
  26.             threads[i] = new JoinThread();  
  27.         for (int i = 0; i < threads.length; i++)  
  28.             // 运行刚才建立的100个线程  
  29.             threads[i].start();  
  30.         for (int i = 0; i < threads.length; i++)  
  31.             // 100个线程都执行完后继续  
  32.             threads[i].join();  
  33.         System.out.println("n=" + JoinThread.n);  
  34.     }  
  35. }  
  36.        

如果对n的操作是原子级别的,最后输出的结果应该为n=1000,而在执行上面积代码时,很多时侯输出的n都小于1000,这说明n=n+1不是原子级别的操作。原因是声明为volatile的简单变量如果当前值由该变量以前的值相关,那么volatile关键字不起作用,也就是说如下的表达式都不是原子操作:

n = n + 1;
n++;

如果要想使这种情况变成原子操作,需要使用synchronized关键字,如上的代码可以改成如下的形式:

  1. package mythread;  
  2.  
  3. public class JoinThread extends Thread  
  4. {  
  5.     public static int n = 0;  
  6.  
  7.     public static synchronized void inc()  
  8.     {  
  9.         n++;  
  10.     }  
  11.     public void run()  
  12.     {  
  13.         for (int i = 0; i < 10; i++)  
  14.             try 
  15.             {  
  16.                 inc(); // n = n + 1 改成了 inc();  
  17.                 sleep(3); // 为了使运行结果更随机,延迟3毫秒  
  18.  
  19.             }  
  20.             catch (Exception e)  
  21.             {  
  22.             }  
  23.     }  
  24.  
  25.     public static void main(String[] args) throws Exception  
  26.     {  
  27.  
  28.         Thread threads[] = new Thread[100];  
  29.         for (int i = 0; i < threads.length; i++)  
  30.             // 建立100个线程  
  31.             threads[i] = new JoinThread();  
  32.         for (int i = 0; i < threads.length; i++)  
  33.             // 运行刚才建立的100个线程  
  34.             threads[i].start();  
  35.         for (int i = 0; i < threads.length; i++)  
  36.             // 100个线程都执行完后继续  
  37.             threads[i].join();  
  38.         System.out.println("n=" + JoinThread.n);  
  39.     }  
  40. }  

上面的代码将n=n+1改成了inc(),其中inc方法使用了synchronized关键字进行方法同步。因此,在使用volatile关键字时要慎重,并不是只要简单类型变量使用volatile修饰,对这个变量的所有操作都是原来操作,当变量的值由自身的上一个决定时,如n=n+1、n++等,volatile关键字将失效,只有当变量的值和自身上一个值无关时对该变量的操作才是原子级别的,如n = m + 1,这个就是原级别的。所以在使用volatile关键时一定要谨慎,如果自己没有把握,可以使用synchronized来代替volatile。

分享到:
评论

相关推荐

    java volatile 关键字实战

    java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java volatile 关键字实战java ...

    volatile关键字使用

    此外,`volatile`关键字的使用需要谨慎,过度使用可能会导致性能下降,因为它阻止了编译器的优化。只有在确实需要处理可能由非程序控制因素更改的变量时,才应该使用`volatile`。在大多数情况下,使用线程安全的数据...

    Java并发编程:volatile关键字解析

    使用`volatile`关键字可以确保在多个线程试图初始化同一个对象时,能够正确地处理可见性和有序性问题。 总结来说,`volatile`关键字是Java并发编程中一个非常重要的概念。它不仅解决了多线程环境下的可见性问题,还...

    详解C中volatile关键字

    volatile关键字的含义是与优化有关,编译器有一种技术叫做数据流分析,分析程序中的变量在哪里赋值、在哪里使用、在哪里失效,分析结果可以用于常量合并、常量传播等优化,进一步可以死代码消除。但有时这些优化不是...

    深入解析Java中的volatile关键字:原理、应用与实践

    本文将详细探讨volatile关键字的工作原理、使用场景以及如何在实际开发中正确使用volatile。 volatile关键字是Java并发编程中一个重要的工具,它通过确保变量的可见性和禁止指令重排序来提高程序的并发性能。然而,...

    volatile的使用方法

    volatile 关键字在 C 语言中的使用方法 在 C 语言中,volatile 关键字是用来指定变量的存储类别的,它告诉编译器这个变量的值可能会在编译器不知道的情况下被修改,因此需要重新加载这个变量的值。volatile 关键字...

    深入多线程之:内存栅栏与volatile关键字的使用分析

    以前我们说过在一些简单的例子中,比如为一个字段赋值或递增该字段,我们需要对线程进行同步,虽然lock可以满足我们的需要,但是一个竞争锁一定会导致阻塞,然后忍受线程上下文切换和调度的开销,在一些高并发和性能...

    AVR-GCC 中如何使用volatile 关键字.pdf

    ### AVR-GCC 中如何使用 `volatile` 关键字 #### 1. `volatile` 关键字的基本概念 在编程语言中,尤其是C/C++中,`volatile` 是一个类型限定符,用来修饰变量,告知编译器该变量的值可能会在编译器未监控的情况下...

    C语言中关于关键字volatile的用法

    这些寄存器的值可能会被外部设备或者中断服务程序改变,因此,在访问这些寄存器时,必须使用volatile关键字进行声明。这样编译器就不会因为编译优化而忽略对外部状态的检查,从而确保每次访问都是直接与硬件交互。 ...

    Java线程:volatile关键字

    Java 线程 volatile 关键字详解 Java™ 语言包含两种内在的同步机制:同步块(或方法)和 volatile 变量。volatile 变量的同步性较差,但它有时更简单并且开销更低。volatile 变量可以被看作是一种 “程度较轻的 ...

    C语言关键字volatile(易变的)表示不经过赋值,其值也可能被改变

    使用volatile关键字可以确保编译器不会对变量进行优化,以确保程序的正确执行。例如,在中断服务程序中,某些变量可能会被修改,以供其他程序检测。在这种情况下,需要使用volatile关键字,以防止编译器对变量进行...

    C++中volatile关键字及常见的误解总结

    C++中volatile关键字及常见的误解总结 C++中的volatile关键字是一种类型修饰符,用来修饰变量,表示该变量可以被某些编译器未知的因素更改,比如操作系统、硬件或者其它线程等。 volatile关键字的主要作用是提供对...

    深入理解 volatile 关键字.doc

    深入理解volatile关键字 volatile关键字是Java语言的高级特性,它可以保证可见性和禁止指令重排序,但是要弄清楚...在使用volatile关键字时,需要遵守相关的规则,并且需要弄清楚Java内存模型,以免出现不必要的错误。

    java里的volatile关键字详解

    "Java中的Volatile关键字详解" Java中的Volatile关键字详解是Java中的一种关键字,用于保证线程之间的可见性、原子性和有序性。下面是对Java中的Volatile关键字详解的知识点总结: 一、基本概念 1. 可见性:可见...

    volatile关键字 Const关键字 static关键字 mutable 关键字

    在C/C++编程语言中,`volatile`, `const`, `static` 和 `mutable` 是...理解和正确使用这些关键字能帮助我们编写更安全、更高效、更具维护性的代码。在实际编程中,应根据具体需求选择合适的关键字,以实现预期的功能。

    关键字volatile的用途

    如果不使用`volatile`关键字,编译器可能会优化代码,将`T`的值加载到寄存器中,从而导致`while`循环成为死循环,因为寄存器中的值不会更新。通过使用`volatile`关键字,确保了每次循环都会从内存中重新读取`T`的值...

    java volatile 关键字 学习

    java volatile 关键字 学习

    一文精通Java中的volatile关键字

    Java中的`volatile`关键字是多线程编程中的一个重要概念,它的主要作用是确保共享变量的可见性和禁止指令重排序。本文将深入探讨`volatile`的关键特性、工作原理以及使用注意事项。 1. 可见性: `volatile`关键字...

    C语言中关键字volatile的作用

    C语言中关键字volatile的作用,使用说明和例子

Global site tag (gtag.js) - Google Analytics