
本文深入探讨了Java多线程环境中对象与引用、堆与栈内存的关系,以及线程如何安全地共享和访问对象。通过阐明引用变量与实际对象实例的区别,并结合Java内存模型(JMM)的“Happens-Before”原则,解释了并发编程中可见性和有序性的挑战。文章还通过具体代码示例分析了安全与不安全的并发场景,并提供了避免常见陷阱的专业指导。
在Java多线程编程中,理解对象、引用、堆内存与栈内存之间的关系是至关重要的。许多初学者常误认为一个对象“属于”创建它的线程,或者当线程进入循环时就无法再与其中声明的对象交互。然而,这种理解并不完全准确。
在Java中,所有对象实例(如通过 new 关键字创建的实例)都存储在堆内存(Heap)中。堆内存是所有线程共享的区域。这意味着,一个对象一旦被创建,它就存在于堆上,任何持有该对象引用的线程都可以访问它,而不管这个引用最初是在哪个线程中获得的。
理解引用变量和对象实例是解决混淆的关键。
立即学习“Java免费学习笔记(深入)”;
考虑以下代码片段:
whatTime wt = new whatTime();
这行代码执行了两个截然不同的操作:
其他线程无法直接访问当前线程的栈内存,因此它们无法直接访问 wt 这个引用变量本身。但是,它们可以获得 wt 所指向的那个堆上的 whatTime 实例的地址副本。
当我们将一个引用变量传递给另一个线程时,Java采用的是值传递(pass-by-value)机制。这意味着传递的是引用变量的副本,而不是引用变量本身。
threadA ta = new threadA(wt); ta.start();
在这段代码中:
现在,mainClass 线程和 threadA 线程都各自持有一个指向同一个 whatTime 对象的引用。它们都拥有“地址簿页面”的副本,这些副本都指向堆上的同一个“房子”。因此,threadA 线程可以随时通过它自己的 wt 引用来调用 whatTime 对象的方法,就像 mainClass 线程也可以一样。
虽然多个线程可以共享和访问同一个对象,但这并不意味着并发访问总是安全的。Java为了提高性能,允许编译器和处理器进行指令重排,并使用CPU缓存。这可能导致在多线程环境下出现可见性(Visibility)和有序性(Ordering)问题。
现代CPU为了提高数据访问速度,会在每个核心内部设置高速缓存(Cache)。当一个线程修改了共享变量的值时,这个修改可能首先写入CPU缓存,而不是立即写入主内存。如果另一个线程从主内存读取这个变量,它可能读取到的是旧值,因为缓存中的新值尚未刷新到主内存。这就是可见性问题。
编译器和处理器为了优化执行效率,可能会对指令进行重排序,只要不改变单线程程序的执行结果。但在多线程环境下,这种重排可能导致一个线程观察到另一个线程的操作顺序与预期不符,从而引发有序性问题。
为了解决这些并发问题,Java内存模型(JMM)引入了“Happens-Before”原则。如果操作A“Happens-Before”操作B,那么操作A的结果对操作B是可见的,并且操作A在操作B之前执行。JMM定义了一系列规则来建立Happens-Before关系,包括:
如果多个线程同时访问并修改同一个共享变量而没有建立适当的Happens-Before关系,程序行为将是不可预测的。
class Example {
int x;
void crazy() {
x = 1;
new Thread(() -> x = 5).start(); // 线程1修改x
new Thread(() -> x = 10).start(); // 线程2修改x
System.out.println(x); // main线程读取x
}
}在上述 crazy() 方法中,main 线程启动了两个新线程,它们都试图修改共享变量 x。由于没有同步机制(如 synchronized 或 volatile),main 线程在打印 x 的值时,可能会打印出 1、5 或 10。甚至,在某些极端的CPU架构和JVM实现下,也可能打印出其他意想不到的值。这种行为是不可预测的,且难以测试和调试。
回顾最初的问题代码:
public class mainClass {
public static void main(String[] args) {
whatTime wt = new whatTime(); // (1)
threadA ta = new threadA(wt); // (2)
ta.start(); // (3)
while (true) {
// main线程进入无限循环
}
}
}
public class threadA extends Thread {
private whatTime wt; // (4)
public threadA(whatTime wt) {
this.wt = wt; // (5)
}
public void run() {
while (true) {
try {
Thread.sleep(10000);
System.out.println("threadA: " + wt.getTime()); // (6)
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
public class whatTime {
public long getTime() {
return System.currentTimeMillis(); // (7)
}
}这段代码是安全的,原因如下:
所以,尽管 main 线程进入了一个 while(true) 循环,这仅仅意味着 main 线程本身在忙碌地执行一个空循环,但它并不妨碍 threadA 线程通过其持有的 wt 引用访问 whatTime 对象。
通过遵循这些原则并深入理解Java内存模型,开发者可以编写出健壮、高效且线程安全的并发应用程序。
以上就是Java多线程中对象与引用的深度解析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号