java变量作用域由声明位置决定:局部变量限于{}内且须初始化;成员变量属对象、有默认值、可被this访问;静态变量属类、共享且早于实例初始化;参数变量为特殊局部变量,调用时自动初始化。

Java变量的作用域完全由声明位置决定,没有额外语法标记——写在哪,就活在哪。
局部变量:只在{}里有效,不初始化就用直接报错
你在方法里、if块里、for循环里声明的变量,比如int i = 0;或String tmp = "test";,作用域就是从声明那行开始,到最近的右大括号}结束。
- 没赋值就用?编译器立刻抛出
variable might not have been initialized - 同一层
{}里重复声明同名变量?编译失败,例如int x = 1; int x = 2; - 嵌套块可以重名,但内层会遮蔽外层:
if (true) { String s = "inner"; } String s = "outer";是合法的,因为两个s不在同一作用域 - 不能加
static、private等修饰符——static int x = 1;在方法里是语法错误
成员变量(实例变量):属于对象,整个类里都能用,有默认值
写在类里、所有方法外面,又没加static的变量,比如String name;或int age;,就是成员变量。它跟着对象走,每个new Person()都有自己的副本。
- 不用初始化也能编译通过:
int count;默认就是0,String str;默认是null - 可在任意非静态方法中直接访问,包括构造器、getter/setter、普通业务方法
- 和局部变量同名时,局部变量优先;想访问成员变量得加
this.name - 被
private修饰后,哪怕在同一个类里,也不能在静态方法中直接访问——static方法没有this
静态变量(类变量):全类共享,加载类时就存在
加了static的成员变量,比如public static final String VERSION = "1.2";,属于类本身,不是某个对象。所有实例共用一份,连没创建对象也能用。
立即学习“Java免费学习笔记(深入)”;
- 推荐用
ClassName.VAR_NAME访问,比如Counter.count,而不是obj.count(后者虽能用,但语义模糊) - 静态方法中可以直接读写,但不能访问非静态的成员变量或方法——因为此时可能根本没对象
- 初始化时机早于任何实例:类第一次被主动使用(如调用静态方法、new 实例、访问静态字段)时触发类加载和静态变量初始化
- 如果被
final修饰且是编译期常量(如字面量字符串、基本类型),会被内联优化,注意多模块部署时版本不一致风险
参数变量:形参也是局部变量,但一进来就已初始化
方法括号里的东西,比如public void setName(String name)中的name,本质就是个特殊局部变量:作用域覆盖整个方法体,且调用时自动完成初始化。
- 它和成员变量同名是最常见遮蔽场景,必须用
this.name = name;来区分 - lambda 表达式的参数也一样,而且不能和所在方法的局部变量重名,否则编译报错
- 重载方法之间参数名互不影响,
void set(String a)和void set(int a)完全没问题 - 别在方法里再声明同名局部变量,比如
void f(String s) { String s = "x"; }—— 这是非法的,编译不过
真正容易翻车的不是“哪里能写”,而是“哪里该用this”“为什么static方法里访问不了name”“为什么for里声明的i在循环外还能用(或不能用)”。这些都得盯着声明位置看——Java 不靠关键字圈范围,就靠{}和类/方法边界吃饭。










