多线程

本文深入探讨了Java中的volatile关键字,解析了其在多线程环境中的使用,包括volatile如何确保数组引用的可见性但无法保证数组元素的原子性。文章还介绍了volatile提供内存屏障的作用,确保数据在多线程间的正确同步。此外,讨论了线程同步的实践,如避免伪共享问题,并提到了线程局部变量和线程安全的单例模式。同时,文章涉及了Java中wait和notify的正确使用,以及如何在多线程环境中避免死锁和内存泄漏的问题。
摘要由CSDN通过智能技术生成

1、Java 中能创建 volatile 数组吗?

能,Java 中可以创建 volatile 类型数组,不过只是一个指向数组的引用,而不

是整个数组。我的意思是,如果改变引用指向的数组,将会受到 volatile 的保护,

但是如果多个线程同时改变数组的元素,volatile 标示符就不能起到之前的保护

作用了。

2、volatile 能使得一个非原子操作变成原子操作吗?

一个典型的例子是在类中有一个 long 类型的成员变量。如果你知道该成员变量

会被多个线程访问,如计数器、价格等,你最好是将其设置为 volatile。为什么?

因为 Java 中读取 long 类型变量不是原子的,需要分成两步,如果一个线程正

在修改该 long 变量的值,另一个线程可能只能看到该值的一半(前 32 位)。

但是对一个 volatile 型的 long 或 double 变量的读写是原子。

3、volatile 修饰符的有过什么实践?

一种实践是用 volatile 修饰 long 和 double 变量,使其能按原子类型来读写。

double 和 long 都是 64 位宽,因此对这两种类型的读是分为两部分的,第一次

读取第一个 32 位,然后再读剩下的 32 位,这个过程不是原子的,但 Java 中

volatile 型的 long 或 double 变量的读写是原子的。volatile 修复符的另一个

作用是提供内存屏障(memory barrier),例如在分布式框架中的应用。简单的

说,就是当你写一个 volatile 变量之前,Java 内存模型会插入一个写屏障(write

barrier),读一个 volatile 变量之前,会插入一个读屏障(read barrier)。意

思就是说,在你写一个 volatile 域时,能保证任何线程都能看到你写的值,同时,

在写之前,也能保证任何数值的更新对所有线程是可见的,因为内存屏障会将其

他所有写的值更新到缓存。

4、volatile 类型变量提供什么保证?

volatile 变量提供顺序和可见性保证,例如,JVM 或者 JIT 为了获得更好的性能

会对语句重排序,但是 volatile 类型变量即使在没有同步块的情况下赋值也不会

与其他语句重排序。 volatile 提供 happens-before 的保证,确保一个线程的

修改能对其他线程是可见的。某些情况下,volatile 还能提供原子性,如读 64 位

数据类型,像 long 和 double 都不是原子的,但 volatile 类型的 double 和

long 就是原子的。

5、10 个线程和 2 个线程的同步代码,哪个更容易写?

从写代码的角度来说,两者的复杂度是相同的,因为同步代码与线程数量是相互

独立的。但是同步策略的选择依赖于线程的数量,因为越多的线程意味着更大的

竞争,所以你需要利用同步技术,如锁分离,这要求更复杂的代码和专业知识。

6、你是如何调用 wait()方法的?使用 if 块还是循环?为什

么?

wait() 方法应该在循环调用,因

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值