Java引用变量存储的是JVM管理的逻辑句柄(reference),非C风格指针;它不可运算、不可解引用,仅用于访问对象,赋值和传参均为该句柄的值拷贝。

Java里的引用类型不是“指向对象的指针”这种C风格的理解,而是**变量保存的是对象在堆内存中的逻辑句柄(reference)**——它不可直接运算、不可解引用,只用于访问对象状态和行为。
引用变量存储的是什么
声明一个引用变量如 String s,JVM会在栈上分配一小块空间存一个reference值;这个值由JVM内部管理,可能是对象地址、句柄、或间接索引,具体取决于虚拟机实现(如HotSpot默认使用直接指针)。你无法获取或修改它,也不能做s + 1这类操作。
- 所有引用类型(
class、interface、array、enum、record)变量都遵循这一规则 - 基本类型(
int、boolean等)变量则直接存值,不涉及堆 -
null是一个特殊的reference值,表示“不指向任何对象”
赋值和传参时发生的是引用拷贝
Java永远是值传递,但对引用类型来说,“值”就是那个reference。所以:
String a = new String("hello");
String b = a; // b 得到的是 a 所持 reference 的副本,二者指向同一对象
b = "world"; // b 现在指向新字符串对象,a 不受影响
- 修改
b本身(如重新赋值)不影响a - 但如果通过
b调用方法改变对象状态(如StringBuilder.append()),a看到的也是同一份数据 - 数组、
ArrayList、自定义对象都一样:变量间传递的是reference拷贝,不是对象拷贝
常见误解与典型错误
很多人以为==比较的是“内容是否相同”,其实它比较的是两个变量持有的reference是否相等(即是否指向同一对象):
立即学习“Java免费学习笔记(深入)”;
String s1 = "abc";
String s2 = "abc";
String s3 = new String("abc");
System.out.println(s1 == s2); // true(字符串常量池优化)
System.out.println(s1 == s3); // false(堆上新对象)
System.out.println(s1.equals(s3)); // true(重写了 equals,比内容)
- 用
==判断两个List是否“内容相同”?错,要用list1.equals(list2) - 认为
Integer i = 1000; Integer j = 1000; i == j一定为true?错,只有-128 ~ 127范围内的Integer才缓存复用 - 把
ArrayList传进方法后清空,原列表也被清空?对——因为传入的是reference拷贝,仍指向原对象
真正容易被忽略的,是JVM对reference的具体实现细节完全不暴露给开发者:你不能假设它是地址、不能强制转型、不能算偏移。它的全部意义,就是让JVM能定位并管理堆上的对象——仅此而已。










