volatile的作用和syncornized的区别

本文通过代码示例探讨了volatile关键字在多线程环境中的作用,揭示其如何解决线程间的可见性问题,同时对比了volatile与synchronized在确保数据可见性和原子性方面的差异。volatile仅能保证变量的可见性,不保证原子性,而synchronized则提供了更全面的线程同步机制。
摘要由CSDN通过智能技术生成

这个是面试高频题

首先上代码看一下volatile的作用
public class JMMTest {

      private boolean flag = false;
    public static void main(String[] args) throws Exception {
        final JMMTest jmmTest = new JMMTest();
        //实现Runable接口 使用匿名内部类简化代码
       new Thread(new Runnable() {
           @Override
           public void run() {
               System.out.println("thread1--start");
               while (!jmmTest.flag){
               }
               System.out.println("thread1--end");
           }
       }).start();
       Thread.sleep(100);
       new Thread(new Runnable() {
           @Override
           public void run() {
               System.out.println("thread2--start");
               jmmTest.flag=true;
               System.out.println("thead2---end");
           }
       }).start();
    }
}

 

上面代码的运行结果是

线程1无法感知到看到线程2对共享变量flag的改变,所以!jmmTest.flag一直是初始值true,所以就无限的循环

当给变量falg加上valotile关键字的时候运行结果是这样的

说明在加volatile之后线程1就能感知到看到线程2对共享变量的改变,然而进行相关的判断操作

 

结论

    //主内存中  解决不同线程之间的可见性   缓存一致性协议   缓存一致性问题  总线嗅探机制的监听

    //1   lock指令 汇编指令
    //2   强制修改的值写入主内存
    //3   当写操作的时候,其它线程对应的cpu缓存暂时失效

volatile的作用和syncornized的区别主要有两点

1:valotile只能作用于变量,而syncornized可以作用于变量和方法

2:volitle只能保证数据的可见性,不能保证原子性,syncornized既可以保证数据的可见性,在多线程下还可以保证原子性(即原子性)

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: "volatile" 是一种 C/C++ 中的关键字,用于告诉编译器该变量可能会在程序执行期间被意外修改,因此编译器在对其进行优化时必须小心谨慎,不能进行一些假设优化。在多线程编程中,使用 "volatile" 关键字可以确保线程之间对变量的修改能够被及时地同步。 举例来说,如果一个变量被声明为 "volatile" 类型,那么编译器在编译代码时就不能对该变量进行某些优化,比如将变量的值缓存在寄存器中而不是从内存中读取,或者将多次读取变量的操作合并成一次操作。因为这些优化可能会导致程序在运行时出现错误,而 "volatile" 的作用就是告诉编译器不要对该变量进行这些优化。 在实际应用中,使用 "volatile" 关键字的场景比较少,因为在大多数情况下,程序员可以通过其他手段来避免变量被意外修改,比如使用锁来保证变量在多线程环境下的原子性访问。因此,如果不是非常确定自己需要使用 "volatile",最好不要随意使用。 ### 回答2: volatileJava中的关键字,用于修饰变量。它的主要作用是确保多线程环境下变量的可见性和有序性。在多线程情况下,各个线程可能会对共享变量进行操作,而volatile可以保证变量的修改能够立即被其他线程看到。 具体来说,当一个变量被volatile修饰时,每个线程在读取这个变量时都会去主内存中获取最新的值,而不是使用自己线程的工作内存。这样可以避免出现某个线程修改了变量的值但其他线程仍然使用旧值的情况。 另外,volatile还可以保证变量的修改操作具有原子性。普通的变量在进行自增、自减等操作时,可能会被切换线程中断,导致最终结果不符合预期。而volatile修饰的变量,其自增、自减等操作是原子的,不会受到线程切换的干扰。 使用volatile的场景包括: 1. 适用于多个线程共享的变量,例如全局变量或静态变量。 2. 适用于变量的值不依赖于当前值的操作,例如标记变量的修改。 需要注意的是,虽然volatile能够保证可见性和有序性,但并不能保证变量的操作是原子的。如果需要保证原子性,应该使用synchronized或atomic类。另外,使用volatile过多会增加系统的开销,因此在合适的情况下使用才是明智的选择。 ### 回答3: volatile 是一个关键字,用于声明变量。它的作用是告知编译器和处理器,在变量被多个线程同时访问时,每次访问该变量都必须从主内存中读取最新的值,而不是使用本地缓存中的旧值。 volatile 的用法有两个方面: 1. 线程间的可见性:当一个变量被多个线程访问时,为了保证变量的可见性,我们可以使用 volatile 关键字。它能够保证在一个线程中修改了该变量的值后,其他线程能够立即看到最新的值。 2. 禁止指令重排序:编译器和处理器为了提高程序的执行效率,可能会对指令进行重排序。然而,在多线程环境下,这种重排序可能引发问题。通过使用 volatile 关键字,我们可以告知编译器和处理器,该变量的读写操作不能被重排序,从而避免了潜在的线程安全问题。 需要注意的是,虽然 volatile 关键字能够保证可见性和禁止重排序,但它并不能解决所有的线程安全问题。例如,volatile 无法保证原子性。如果需要保证原子性,应该考虑使用 synchronized 关键字或 Lock 接口中的相应方法。 总结起来,volatile 关键字的作用是确保变量在多线程环境下的可见性和禁止指令重排序。它适用于那些被多个线程共享且可能同时修改的变量。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值