局部变量在方法内定义,作用域限于声明它的大括号内,方法结束即释放;成员变量属于类或实例,有默认初始值,可被修饰符修饰,同名时局部变量遮蔽成员变量。

局部变量在方法内定义,离开作用域就不可访问
局部变量写在方法、构造器或代码块内部,比如 public void doWork() { int count = 0; } 中的 count。它只在声明它的那对大括号 {} 内有效,方法执行完后内存立即释放,无法被其他方法读取。
常见错误是试图在方法外访问它:
public void doWork() {
int x = 10;
}
public void printX() {
System.out.println(x); // 编译报错:cannot resolve symbol 'x'
}
- 不能用
static、public等修饰符修饰局部变量 - 未显式初始化就使用会编译失败(如
int y; System.out.println(y);) - 同名局部变量会遮蔽同名成员变量,需用
this.x显式访问成员变量
成员变量属于类或实例,生命周期与对象绑定
成员变量定义在类内部、方法外部,分两种:static 成员变量(类变量)和非 static 成员变量(实例变量)。前者随类加载而存在,后者随对象创建而分配内存,对象销毁时才可能被回收。
它们有默认初始值:int 是 0,boolean 是 false,引用类型是 null,无需手动初始化就能直接读取(但读 null 可能引发 NullPointerException)。
-
static成员变量被所有实例共享,修改一个对象的该变量会影响其他对象 - 非
static成员变量每个对象一份独立副本 - 可使用
private、protected、public、static、final等修饰符
同名变量冲突时,局部变量优先级高于成员变量
当方法内定义了和成员变量同名的局部变量,所有未加限定的引用都指向局部变量。这是遮蔽(shadowing),不是覆盖(overriding)。
立即学习“Java免费学习笔记(深入)”;
class Example {
private int value = 100;
public void show() {
int value = 200; // 局部变量遮蔽成员变量
System.out.println(value); // 输出 200
System.out.println(this.value); // 输出 100,显式访问成员变量
}
}
- 遮蔽只发生在局部变量与同名成员变量之间,不适用于继承中的同名字段(子类字段也会遮蔽父类字段)
- IDE 通常会警告这种命名方式,建议避免用相同名字降低可读性
- 若想在局部作用域里明确操作成员变量,必须用
this.(实例)或ClassName.(静态)前缀
编译期检查严格区分二者,运行时行为也完全不同
Java 编译器在编译阶段就确定变量是否可访问:局部变量查作用域嵌套,成员变量查访问修饰符+继承关系。一旦编译通过,局部变量存于栈帧的局部变量表,成员变量存于堆中对象实例或方法区的类结构里。
- 局部变量过多或过大可能引发
StackOverflowError(尤其递归深度大时) - 大量未及时置为
null的成员变量引用,可能阻碍 GC 回收对象,造成内存泄漏 - 多线程环境下,局部变量天然线程安全;而成员变量(尤其是可变的)需额外同步机制
最容易被忽略的是遮蔽带来的逻辑误判——你以为改的是成员变量,其实只是局部副本;或者忘了 this. 导致始终在操作局部值。调试时注意变量作用域高亮和 IDE 提示。










