多线程(56)final关键字在Java内存模型中的作用

本文详细阐述了Java中final关键字的用法,尤其是在内存模型(JMM)中的作用,强调了final字段的初始化安全性以及其在创建线程安全不可变对象中的重要性。同时讨论了内存屏障在final字段写入过程中的角色和注意事项。
摘要由CSDN通过智能技术生成

在Java中,final关键字可以用于变量、方法和类。当用于变量时,final确保变量一旦被初始化后其值就不能被改变;用于方法时,确保方法不被重写;用于类时,确保类不被继承。

对于Java内存模型(JMM)来说,final字段的主要作用是提供初始化安全性,确保对象一旦被构造完成,其final字段的值就不会被改变,并且对于任何获取这些字段的线程来说,final字段的值总是一致的。

final字段的内存语义

在JMM中,写入final字段的重排序规则确保了构造函数内对一个final字段的写入,与随后把这个被构造对象的引用赋值给一个引用变量,这两个动作不能被重排序。这种重排序会在构造函数内对final字段的所有写入操作之后执行,这意味着对象的引用直到final字段被正确初始化之后才能从构造函数中逸出。

代码演示

public class FinalFieldExample {
    final int x;
    int y;
    static FinalFieldExample f;

    public FinalFieldExample() {
        x = 3;  // 对final字段的写入
        y = 4;  // 对普通字段的写入
    }

    static void writer() {
        f = new FinalFieldExample();
    }

    static void reader() {
        if (f != null) {
            int i = f.x;  // 读操作
            int j = f.y;  // 读操作
        }
    }
}

在这个示例中,一旦FinalFieldExample对象通过构造函数完全构造后,任何线程都可以保证看到final字段x的值为3,而无论是否有同步。但是对于普通字段y,如果没有适当的同步,其他线程可能看不到初始化后的值。

初始化安全性

对于包含final字段的对象,JMM提供了一种特殊的保证,称为初始化安全性。这意味着,对于被正确构造的(构造过程中没有this逸出)不可变对象,其他线程都能看到final字段在构造器中设置的值,即使没有同步措施,这个保证也是有效的。

初始化安全性可以保障在对象构造完成后,任何线程访问对象的final字段都将获取到与构造器中一样的值。

源码层面解析

在JDK的源码中,final字段的特殊处理主要体现在JVM的内存屏障插入策略中。在构造函数中写入final字段时,JVM会在构造函数末尾插入一个StoreStore屏障,确保final字段的赋值操作不会被重排序到构造函数之外。

同时,初始化安全性的保证意味着当构造函数完成时,所有对final字段的写入都将在任何后续读取这些字段之前发生内存屏障。

注意事项

  • final字段一旦被初始化后,其语义保证了不可变性,这对于创建不可变对象至关重要。
  • 初始化安全性确保只要对象被正确构造,所有线程都能看到final字段在构造器中被设置的值,即使没有额外的同步措施。
  • 尽管final字段对可见性提供了保证,但它并不能替代所有同步需要。对于涉及多个字段的复杂状态的对象,你可能仍然需要使用synchronizedjava.util.concurrent包中的同步机制。

final字段的使用是创建线程安全的不可变对象的基础,并且可以在没有锁的情况下减少同步的需求,从而提高并发性能。然而,设计线程安全的共享对象依然需要仔细考量,并且通常需要多种同步机制的结合使用。

  • 6
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

辞暮尔尔-烟火年年

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值