# Java里面volatile关键字修饰引用变量的陷阱
如果我现在问你volatile的关键字的作用,你可能会回答对于一个线程修改的变量对其他的线程立即可见。这种说法没多大问题,但是不够严谨。
严谨的回答应该是volatile关键字对于基本类型的修改可以在随后对多个线程的读保持一致,但是对于引用类型如数组,实体bean,仅仅保证引用的可见性,但并不保证引用内容的可见性。
下面这些数据结构都属于引用类型,即使使用volatile关键字修饰,也不能保证修改后的数据会立即对其他的多个线程保持一致:
```java
volatile int [] data;
valatile boolean [] flags;
volatile Person person;
```
如何证明?看下面的一段代码:
```java
private static volatile Data data;
public static void setData(int a, int b) {
data = new Data(a, b);
}
private static class Data {
private int a;
private int b;
public Data(int a, int b) {
this.a = a;
this.b = b;
}
public int getA() {
return a;
}
public int getB() {
return b;
}
}
public static void main(String[] args) throws InterruptedException {
for (int i = 0; i < 10000; i++) {
int a = i;
int b = i;
//writer
Thread writerThread = new Thread(() -> {setData(a, b);});
//reader
Thread readerThread = new Thread(() -> {
while (data == null) {}
int x = data.getA();
int y = data.getB();
if (x != y) {
System.out.printf("a = %s, b = %s%n", x, y);
}
});
writerThread.start();
readerThread.start();
writerThread.join();
readerThread.join();
}
System.out.println("finished");
}
```
上面的代码,有个实体类Data,它有两个字段,分别是a和b,然后在我们的main方法中,我们声明了
一个for循环1万次,在循环体里面我们先声明了一个写入线程,每次给实体类赋值,接着又声明了一个读取线程,当实体不为null的时候,打印如果有不一致的时候,其字段的值。接着同时启动两个线程,并在主线程中分别等待其结束。
在我的mac系统上,运行了第三次的时候出现了不一致:
```
a = 2760, b = 2761
a = 3586, b = 3587
finished
```
原因是对于属性a和b我们都是分别的读取,所以缺乏了happens-before关系的约束。
如何解决这种情况?
(1)去掉独立的getA和getB方法,使用int数组,一次返回两个属性
```
public int[] getValues() {
return new int[]{a, b};
}
```
(2)使用java并发包下面的基于CAS的原子结构:
AtomicReference
```
//修改1
private static AtomicReference<Data> data = new AtomicReference<>();
//修改2
public static void setData(int a, int b) {
data.compareAndSet(null, new Data(a, b));
}
//修改3
Thread readerThread = new Thread(() -> {
while (data.get() == null) {}
int x = data.get().getA();
int y = data.get().getB();
if (x != y) {
System.out.printf("a = %s, b = %s%n", x, y);
}
});
```
总结:
本篇文章主要讲述了关于volatile修饰引用变量的问题即它只能保证引用本身的可见性,并不能保证内部字段的可见性,如果想要保证内部字段的可见性最好使用CAS的数据结构,这里还需要说明的的一点是volatile有时候修饰引用类型如boolean数组可能结果是没问题的,大家可以看我在Stack Overflow上提问的一个问题:
https://stackoverflow.com/questions/50967448/about-java-volatile-array
在编程的世界里面,对于不确定的事情,我们始终都要以最坏的打算来看待,所以请记住:尽量避免使用volatile关键字修饰引用变量。
分享到:
相关推荐
7. **修饰符关键字**:`final` 用于创建不可变对象或常量,`abstract` 用于声明抽象类或方法,`static` 创建静态成员,`volatile` 确保多线程环境下变量的可见性,`synchronized` 用于线程同步。 8. **类型关键字**...
- **demo9**:可能通过volatile关键字实现线程间变量的即时通信。 - **demo5**:可能对比了synchronized和volatile在解决线程安全问题上的差异。 - **demo12**:可能探讨了volatile的使用限制,如非原子性操作的问题...
4. **volatile关键字**: `volatile`关键字用于修饰变量,确保多线程环境下的可见性和有序性。当一个变量被声明为volatile时,它的修改会立即对其他线程可见,防止出现数据不一致的情况。但是,volatile不能保证原子...
JMM为了保证内存可见性,引入了volatile关键字。volatile变量的修改会立即同步到主内存,且读取时会直接从主内存获取,不会使用工作内存的副本。这使得其他线程可以及时看到该变量的最新值,避免数据不一致。 其次...
`synchronized`关键字、`volatile`变量和`java.util.concurrent`包的使用是多线程陷阱的重点。 5. **类与对象**:类的继承、封装和多态是面向对象编程的基础。理解构造函数、访问修饰符和抽象类的使用,以及`final`...
其次,书中深入探讨了Java内存模型(JMM)和volatile关键字。Java内存模型规定了线程如何访问和修改共享变量,以确保多线程环境下的可见性和一致性。volatile保证了变量对所有线程的可见性,并禁止指令重排序,但不...
`volatile`关键字用于修饰变量,确保所有线程都能看到共享变量的最新值。然而,它的使用需要谨慎,因为它仅提供可见性保证,而不保证原子性。 5. 向线程传递数据: 通常,线程间数据传递可以通过类成员变量、线程...
8. **volatile 关键字**: 它确保了变量的可见性和有序性,但不保证原子性。当多个线程访问同一变量时,volatile能保证每个线程都能看到最新的值。 9. **synchronized 关键字**: 它用于保证同一时间只有一个线程可以...
这包括对volatile变量的定义、线程同步(synchronized关键字)以及JVM内存区域的划分。 6. **Java虚拟机(JVM)**:虽然JLS主要关注语言层面,但JVM规范也是理解Java运行时环境的关键。JVM负责解释执行字节码、垃圾...
3. **线程安全**:线程安全是并发编程中的重要主题,书中会探讨各种线程安全的实现策略,如使用`synchronized`关键字、`volatile`变量、`final`修饰以及线程局部变量(`ThreadLocal`)。 4. **并发集合**:Java的...
- `final`:修饰变量或方法,表示不可变或不允许重写。 - `finally`:在`try-catch`块中,即使发生异常也会执行的代码块。 - `finalize`:对象被垃圾回收前可能调用的方法,用于资源清理,但在Java 9之后已过时。 #...