Java 内存模型概述
Java 内存模型(JMM)定义了 Java 虚拟机(JVM)如何处理多线程对共享内存的访问。JMM 规定了线程对共享变量的读写操作必须是原子的,即一个线程对共享变量的修改操作要么全部执行,要么完全不执行,不会出现部分执行的情况。
原子操作
原子操作是指一个不可中断的操作,要么全部执行,要么完全不执行。在 Java 中,原子操作包括:
- 读取或写入一个 volatile 变量
- 读取或写入一个 final 变量
- 读写一个 long 或 double 变量(需要使用 synchronized 关键字或 volatile 关键字)
- 使用 synchronized 方法或代码块对共享变量进行访问
内存屏障
内存屏障是一种特殊的指令,用于强制编译器和处理器按照特定的顺序执行指令。内存屏障可以确保一个线程对共享变量的修改操作对其他线程可见。
在 Java 中,内存屏障可以分为以下几种类型:
- LoadStore 屏障:用于确保一个线程对共享变量的修改操作对其他线程可见。
- StoreLoad 屏障:用于确保一个线程能够看到其他线程对共享变量的修改操作。
- Full 屏障:用于确保一个线程能够看到其他线程对共享变量的所有修改操作。
volatile 变量
volatile 变量是一种特殊的变量,用于确保对该变量的修改操作对所有线程都是可见的。volatile 变量的读写操作都是原子的,并且具有内存屏障的功能。
示例代码
以下示例代码演示了如何使用原子操作、内存屏障和 volatile 变量来确保多线程编程中的原子性和内存可见性:
public class AtomicCounter {
private volatile int count;
public void increment() {
count++;
}
public int getCount() {
return count;
}
}
public class Main {
public static void main(String[] args) {
AtomicCounter counter = new AtomicCounter();
Thread thread1 = new Thread(() -> {
for (int i = 0; i < 100000; i++) {
counter.increment();
}
});
Thread thread2 = new Thread(() -> {
for (int i = 0; i < 100000; i++) {
counter.increment();
}
});
thread1.start();
thread2.start();
try {
thread1.join();
thread2.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("Final count: " + counter.getCount());
}
}
在上面的示例代码中,AtomicCounter 类使用 volatile 变量来确保 count 变量的修改操作对所有线程都是可见的。increment() 方法使用 synchronized 关键字来确保对 count 变量的修改操作是原子的。getCount() 方法使用 volatile 变量来确保对 count 变量的读取操作是可见的。
结论
Java 内存模型(JMM)定义了多线程编程中共享内存的访问规则,以及如何确保原子操作和内存可见性。本文探讨了 JMM 的核心概念,包括原子操作、内存屏障和 volatile 变量,并通过示例代码演示了它们的使用。