• 博客园logo
  • 会员
  • 众包
  • 新闻
  • 博问
  • 闪存
  • 赞助商
  • HarmonyOS
  • Chat2DB
    • 搜索
      所有博客
    • 搜索
      当前博客
  • 写随笔 我的博客 短消息 简洁模式
    用户头像
    我的博客 我的园子 账号设置 会员中心 简洁模式 ... 退出登录
    注册 登录
思想人生从关注生活开始
博客园    首页    新随笔    联系   管理    订阅  订阅

volatile关键字

volatile定义

volatile是Java中的一个关键字,它用于确保变量在多线程环境下的可见性和有序性。当一个字段被声明为volatile时,它具有以下特性:

  1. 可见性:在多线程环境中,一个线程修改了一个volatile变量的值,这个新值对其他线程来说是立即可见的。这意味着当一个线程更新了一个volatile变量,其他线程读取这个变量时会看到最新的值。
  2. 有序性:volatile关键字可以防止指令重排序优化。编译器和处理器为了提高性能,常常会对指令顺序进行重排序。但volatile通过插入内存屏障来保证程序执行的顺序按照代码的先后顺序执行。

通常使用场景

  • 变量的写入操作不依赖变量的当前值,或者只有单个线程修改变量的值,但是其他线程需要立即看到修改后的值。
  • 变量不会被多个线程同时修改,但是会被多个线程读取。
  • 使用volatile关键字保证线程之间的通信,例如使用volatile关键字声明的标识位来控制线程的启停。

需要注意的是,volatile关键字虽然可以保证数据的可见性和有序性,但不能保证线程安全。如果需要进行线程安全的操作,还需要使用其他的线程同步机制,如synchronized等。

在使用volatile时,应谨慎考虑其具体使用场景,并确保理解其工作原理以及可能带来的限制。

精彩示例

示例1

比如我们常见的javax.net.SocketFactory 类的getDefault()获取默认实例(这种单例模型)在大多数情况下是线程安全的,但有改进空间:

改进前

public abstract class SocketFactory {
private static SocketFactory theFactory;

protected SocketFactory() {
}

public static SocketFactory getDefault() {

synchronized(SocketFactory.class) {
if (theFactory == null) {
theFactory = new DefaultSocketFactory();
}
}

return theFactory;
}

   ...}


改进后

在这个改进的版本中,我们仍然使用了同步块来确保theFactory只被创建一次,但是我们还添加了一个volatile关键字来确保theFactory的可见性,并且使用了两次检查(尽管这不是严格的双重检查锁定,因为volatile已经足够确保线程安全)。这样,我们就避免了不必要的同步开销,同时仍然保持了线程安全性。

private static volatile SocketFactory theFactory;

private SocketFactory() {
// 私有构造函数,防止外部类实例化
}

public static SocketFactory getDefault() {
if (theFactory == null) { // 第一次检查实例是否已经被创建
synchronized (SocketFactory.class) {
if (theFactory == null) { // 第二次检查实例是否已经被创建(在同步块内)
theFactory = new DefaultSocketFactory();
}
}
}
return theFactory;
}
   ...}

posted @ 2024-04-24 14:03  JackYang  阅读(16)  评论(0)    收藏  举报
刷新页面返回顶部
博客园  ©  2004-2025
浙公网安备 33010602011771号 浙ICP备2021040463号-3