volatile和synchronize的区别

本文对比了Java中的volatile和synchronized关键字,阐述了它们如何处理执行控制和内存可见性问题,包括synchronized的锁机制、内存屏障效果,以及volatile原子性与可见性的区别,以及各自的适用场景和限制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、前言

首先需要理解线程安全的两个方面:执行控制和内存可见。

  1. 执行控制:目的是控制代码执行(顺序)及是否可以并发执行。
  2. 内存可见:控制的是线程执行结果在内存中对其它线程的可见性。根据Java内存模型的实现,线程在具体执行时,会先拷贝主存数据到线程本地空间中,操作完成后再把结果从线程本地刷到主存。
二、volatile和synchronized特点
  • synchronized关键字解决的是执行控制的问题,它会阻止其它线程获取当前对象的监控锁,这样就使得当前对象中被synchronized关键字保护的代码块无法被其它线程访问,也就无法并发执行。更重要的是,synchronized还会创建一个内存屏障,内存屏障指令保证了所有CPU操作结果都会直接刷到主存中,从而保证了操作的内存可见性。

  • volatile关键字解决的是内存可见性的问题,会使得所有对volatile变量的读写都会直接刷到主存,这样就能满足一些对变量可见性有要求而对读取顺序没有要求的需求,使用volatile关键字仅能实现对原始变量(如boolen、 short 、int 、long等)操作的原子性。

    • 但需要特别注意, volatile不能保证复合操作的原子性,即使只是i++,实际上也是由多个原子操作组成,假如多个线程同时执行i++,volatile只能保证他们操作的i是同一块内存,但依然可能出现写入脏数据的情况。
    • 对于volatile关键字,当且仅当满足以下所有条件时可使用:
    1. 对变量的写入操作不依赖变量的当前值,或者你能确保只有单个线程更新变量的值
    2. 该变量没有包含在具有其他变量的不变式中
三、volatile和synchronized的区别
  1. volatile本质是在告诉jvm当前变量在寄存器(工作内存)中的值是不确定的,需要从主存中读取; synchronized则是锁定当前变量,只有当前线程可以访问该变量,其他线程被阻塞住。
  2. volatile仅能使用在变量级别;synchronized则可以使用在变量、方法、和类级别的
  3. volatile仅能实现变量的修改可见性,不能保证原子性;而synchronized则可以保证变量的修改可见性和原子性
  4. volatile不会造成线程的阻塞;synchronized可能会造成线程的阻塞。
  5. volatile标记的变量不会被编译器优化;synchronized标记的变量可以被编译器优化

参考文章: volatile和synchronize的区别

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值